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
26
.travis.yml
26
.travis.yml
@ -1,27 +1,21 @@
|
|||||||
language: php
|
language: php
|
||||||
|
|
||||||
php:
|
php:
|
||||||
- 5.3
|
- 5.3
|
||||||
- 5.4
|
|
||||||
|
|
||||||
env:
|
env:
|
||||||
- DB=MYSQL CORE_RELEASE=master
|
- DB=MYSQL CORE_RELEASE=master
|
||||||
- DB=PGSQL CORE_RELEASE=master
|
|
||||||
- DB=SQLITE3 CORE_RELEASE=master
|
|
||||||
- PHPCS=1 CORE_RELEASE=master
|
|
||||||
|
|
||||||
matrix:
|
matrix:
|
||||||
exclude:
|
include:
|
||||||
- php: 5.4
|
- php: 5.3
|
||||||
env: DB=PGSQL CORE_RELEASE=master
|
env: DB=PGSQL CORE_RELEASE=master
|
||||||
|
- php: 5.3
|
||||||
|
env: DB=SQLITE CORE_RELEASE=master
|
||||||
- php: 5.4
|
- php: 5.4
|
||||||
env: DB=SQLITE3 CORE_RELEASE=master
|
env: DB=MYSQL CORE_RELEASE=master
|
||||||
- php: 5.4
|
- php: 5.5
|
||||||
env: PHPCS=1 CORE_RELEASE=master
|
env: DB=MYSQL CORE_RELEASE=master
|
||||||
allow_failures:
|
|
||||||
- env: DB=PGSQL CORE_RELEASE=master
|
|
||||||
- env: DB=SQLITE3 CORE_RELEASE=master
|
|
||||||
- env: PHPCS=1 CORE_RELEASE=master
|
|
||||||
|
|
||||||
before_script:
|
before_script:
|
||||||
- phpenv rehash
|
- phpenv rehash
|
||||||
|
@ -12,7 +12,7 @@ and [installation from source](http://doc.silverstripe.org/framework/en/installa
|
|||||||
|
|
||||||
## Bugtracker ##
|
## 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).
|
Please read our [issue reporting guidelines](http://doc.silverstripe.org/framework/en/misc/contributing/issues).
|
||||||
|
|
||||||
## Development and Contribution ##
|
## Development and Contribution ##
|
||||||
|
@ -16,6 +16,12 @@ class CMSBatchActionHandler extends RequestHandler {
|
|||||||
'$BatchAction/confirmation' => 'handleConfirmation',
|
'$BatchAction/confirmation' => 'handleConfirmation',
|
||||||
'$BatchAction' => 'handleBatchAction',
|
'$BatchAction' => 'handleBatchAction',
|
||||||
);
|
);
|
||||||
|
|
||||||
|
private static $allowed_actions = array(
|
||||||
|
'handleBatchAction',
|
||||||
|
'handleApplicablePages',
|
||||||
|
'handleConfirmation',
|
||||||
|
);
|
||||||
|
|
||||||
protected $parentController;
|
protected $parentController;
|
||||||
|
|
||||||
|
@ -6,6 +6,7 @@ class CMSProfileController extends LeftAndMain {
|
|||||||
private static $menu_title = 'My Profile';
|
private static $menu_title = 'My Profile';
|
||||||
|
|
||||||
private static $required_permission_codes = false;
|
private static $required_permission_codes = false;
|
||||||
|
|
||||||
private static $tree_class = 'Member';
|
private static $tree_class = 'Member';
|
||||||
|
|
||||||
public function getResponseNegotiator() {
|
public function getResponseNegotiator() {
|
||||||
|
@ -76,7 +76,7 @@ class LeftAndMain extends Controller implements PermissionProvider {
|
|||||||
* @config
|
* @config
|
||||||
* @var string
|
* @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
|
* @var array
|
||||||
|
@ -460,7 +460,8 @@ body.cms { overflow: hidden; }
|
|||||||
.cms-add-form .step-label { opacity: 0.9; }
|
.cms-add-form .step-label { opacity: 0.9; }
|
||||||
.cms-add-form .step-label .flyout { height: 17px; padding-top: 5px; }
|
.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 .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; }
|
.cms-add-form .parent-mode { padding: 8px; overflow: auto; }
|
||||||
|
|
||||||
#Form_AddForm_PageType_Holder ul { padding-left: 20px; }
|
#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 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 .middleColumn { margin-top: 16px; margin-left: 184px; }
|
||||||
.htmleditorfield-mediaform .htmleditorfield-from-cms .ss-uploadfield .field.treedropdown { border-bottom: 0; padding: 0; }
|
.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) -------------------------------------------- */
|
/** -------------------------------------------- Search forms (used in AssetAdmin, ModelAdmin, etc) -------------------------------------------- */
|
||||||
.cms-search-form { margin-bottom: 16px; }
|
.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, .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 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-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, .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 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; }
|
.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 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 .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-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, .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 #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; }
|
.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-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.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, .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: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 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; }
|
.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($) {
|
(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.
|
// 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
|
// setup jquery.entwine
|
||||||
$.entwine.warningLevel = $.entwine.WARN_LEVEL_BESTPRACTISE;
|
$.entwine.warningLevel = $.entwine.WARN_LEVEL_BESTPRACTISE;
|
||||||
@ -136,7 +149,7 @@ jQuery.noConflict();
|
|||||||
this._super();
|
this._super();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialize layouts
|
// Initialize layouts
|
||||||
this.redraw();
|
this.redraw();
|
||||||
|
|
||||||
@ -228,15 +241,15 @@ jQuery.noConflict();
|
|||||||
},
|
},
|
||||||
this.getLayoutOptions()
|
this.getLayoutOptions()
|
||||||
));
|
));
|
||||||
|
|
||||||
// Trigger layout algorithm once at the top. This also lays out children - we move from outside to
|
// Trigger layout algorithm once at the top. This also lays out children - we move from outside to
|
||||||
// inside, resizing to fit the parent.
|
// inside, resizing to fit the parent.
|
||||||
this.layout();
|
this.layout();
|
||||||
|
|
||||||
// Redraw on all the children that need it
|
// Redraw on all the children that need it
|
||||||
this.find('.cms-panel-layout').redraw();
|
this.find('.cms-panel-layout').redraw();
|
||||||
this.find('.cms-content-fields[data-layout-type]').redraw();
|
this.find('.cms-content-fields[data-layout-type]').redraw();
|
||||||
this.find('.cms-edit-form[data-layout-type]').redraw();
|
this.find('.cms-edit-form[data-layout-type]').redraw();
|
||||||
this.find('.cms-preview').redraw();
|
this.find('.cms-preview').redraw();
|
||||||
this.find('.cms-content').redraw();
|
this.find('.cms-content').redraw();
|
||||||
},
|
},
|
||||||
@ -590,7 +603,7 @@ jQuery.noConflict();
|
|||||||
if(typeof(window.sessionStorage)=="undefined" || window.sessionStorage === null) return;
|
if(typeof(window.sessionStorage)=="undefined" || window.sessionStorage === null) return;
|
||||||
|
|
||||||
var selectedTabs = [], url = this._tabStateUrl();
|
var selectedTabs = [], url = this._tabStateUrl();
|
||||||
this.find('.cms-tabset,.ss-tabset').each(function(i, el) {
|
this.find('.cms-tabset,.ss-tabset').each(function(i, el) {
|
||||||
var id = $(el).attr('id');
|
var id = $(el).attr('id');
|
||||||
if(!id) return; // we need a unique reference
|
if(!id) return; // we need a unique reference
|
||||||
if(!$(el).data('tabs')) return; // don't act on uninit'ed controls
|
if(!$(el).data('tabs')) return; // don't act on uninit'ed controls
|
||||||
@ -650,8 +663,8 @@ jQuery.noConflict();
|
|||||||
} else if(sessionStates) {
|
} else if(sessionStates) {
|
||||||
$.each(sessionStates, function(i, sessionState) {
|
$.each(sessionStates, function(i, sessionState) {
|
||||||
if(tabset.is('#' + sessionState.id)) index = sessionState.selected;
|
if(tabset.is('#' + sessionState.id)) index = sessionState.selected;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if(index !== null) tabset.tabs('select', index);
|
if(index !== null) tabset.tabs('select', index);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
@ -671,7 +684,7 @@ jQuery.noConflict();
|
|||||||
} else {
|
} else {
|
||||||
for(var i=0;i<s.length;i++) {
|
for(var i=0;i<s.length;i++) {
|
||||||
if(s.key(i).match(/^tabs-/)) s.removeItem(s.key(i));
|
if(s.key(i).match(/^tabs-/)) s.removeItem(s.key(i));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -818,18 +831,18 @@ jQuery.noConflict();
|
|||||||
$('.cms-content .Actions').entwine({
|
$('.cms-content .Actions').entwine({
|
||||||
onmatch: function() {
|
onmatch: function() {
|
||||||
this.find('.ss-ui-button').click(function() {
|
this.find('.ss-ui-button').click(function() {
|
||||||
var form = this.form;
|
var form = this.form;
|
||||||
|
|
||||||
// forms don't natively store the button they've been triggered with
|
// forms don't natively store the button they've been triggered with
|
||||||
if(form) {
|
if(form) {
|
||||||
form.clickedButton = this;
|
form.clickedButton = this;
|
||||||
// Reset the clicked button shortly after the onsubmit handlers
|
// Reset the clicked button shortly after the onsubmit handlers
|
||||||
// have fired on the form
|
// have fired on the form
|
||||||
setTimeout(function() {
|
setTimeout(function() {
|
||||||
form.clickedButton = null;
|
form.clickedButton = null;
|
||||||
}, 10);
|
}, 10);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
this.redraw();
|
this.redraw();
|
||||||
this._super();
|
this._super();
|
||||||
@ -974,13 +987,13 @@ jQuery.noConflict();
|
|||||||
$(".cms-search-form button[type=reset], .cms-search-form input[type=reset]").entwine({
|
$(".cms-search-form button[type=reset], .cms-search-form input[type=reset]").entwine({
|
||||||
onclick: function(e) {
|
onclick: function(e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
var form = $(this).parents('form');
|
var form = $(this).parents('form');
|
||||||
|
|
||||||
form.clearForm();
|
form.clearForm();
|
||||||
form.find(".dropdown select").prop('selectedIndex', 0).trigger("liszt:updated"); // Reset chosen.js
|
form.find(".dropdown select").prop('selectedIndex', 0).trigger("liszt:updated"); // Reset chosen.js
|
||||||
form.submit();
|
form.submit();
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1051,7 +1064,7 @@ jQuery.noConflict();
|
|||||||
},
|
},
|
||||||
redrawTabs: function() {
|
redrawTabs: function() {
|
||||||
this.rewriteHashlinks();
|
this.rewriteHashlinks();
|
||||||
|
|
||||||
var id = this.attr('id'), activeTab = this.find('ul:first .ui-tabs-active');
|
var id = this.attr('id'), activeTab = this.find('ul:first .ui-tabs-active');
|
||||||
|
|
||||||
if(!this.data('uiTabs')) this.tabs({
|
if(!this.data('uiTabs')) this.tabs({
|
||||||
|
@ -153,7 +153,7 @@
|
|||||||
iframe.bind('load', function(e) {
|
iframe.bind('load', function(e) {
|
||||||
if($(this).attr('src') == 'about:blank') return;
|
if($(this).attr('src') == 'about:blank') return;
|
||||||
|
|
||||||
$(this).show();
|
iframe.addClass('loaded').show(); // more reliable than 'src' attr check (in IE)
|
||||||
self._resizeIframe();
|
self._resizeIframe();
|
||||||
self.uiDialog.removeClass('loading');
|
self.uiDialog.removeClass('loading');
|
||||||
}).hide();
|
}).hide();
|
||||||
@ -170,7 +170,7 @@
|
|||||||
var self = this, iframe = this.element.children('iframe');
|
var self = this, iframe = this.element.children('iframe');
|
||||||
|
|
||||||
// Load iframe
|
// Load iframe
|
||||||
if(!iframe.attr('src') || this.options.reloadOnOpen) {
|
if(this.options.iframeUrl && (!iframe.hasClass('loaded') || this.options.reloadOnOpen)) {
|
||||||
iframe.hide();
|
iframe.hide();
|
||||||
iframe.attr('src', this.options.iframeUrl);
|
iframe.attr('src', this.options.iframeUrl);
|
||||||
this.uiDialog.addClass('loading');
|
this.uiDialog.addClass('loading');
|
||||||
@ -186,7 +186,7 @@
|
|||||||
$(window).unbind('resize.ssdialog');
|
$(window).unbind('resize.ssdialog');
|
||||||
},
|
},
|
||||||
_resizeIframe: function() {
|
_resizeIframe: function() {
|
||||||
var opts = {}, newWidth, newHeight;
|
var opts = {}, newWidth, newHeight, iframe = this.element.children('iframe');;
|
||||||
if(this.options.widthRatio) {
|
if(this.options.widthRatio) {
|
||||||
newWidth = $(window).width() * this.options.widthRatio;
|
newWidth = $(window).width() * this.options.widthRatio;
|
||||||
if(this.options.minWidth && newWidth < this.options.minWidth) {
|
if(this.options.minWidth && newWidth < this.options.minWidth) {
|
||||||
@ -207,11 +207,26 @@
|
|||||||
opts.height = newHeight;
|
opts.height = newHeight;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if(this.options.autoPosition) {
|
|
||||||
opts.position = this.options.position;
|
|
||||||
}
|
|
||||||
if(!jQuery.isEmptyObject(opts)) {
|
if(!jQuery.isEmptyObject(opts)) {
|
||||||
this._setOptions(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 {
|
ul.SelectionGroup {
|
||||||
padding-left:28px;
|
padding-left:28px;
|
||||||
|
overflow: visible;
|
||||||
|
@include legacy-pie-clearfix;
|
||||||
}
|
}
|
||||||
.parent-mode {
|
.parent-mode {
|
||||||
padding: $grid-x;
|
padding: $grid-x;
|
||||||
@ -1533,9 +1535,14 @@ body.cms-dialog {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.ss-uploadfield-editandorganize {
|
.ss-assetuploadfield .ss-uploadfield-editandorganize {
|
||||||
display: none;
|
.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']) {
|
if(!isset($databaseConfig) || !isset($databaseConfig['database']) || !$databaseConfig['database']) {
|
||||||
echo "\nPlease configure your database connection details. You can do this by creating a file
|
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";
|
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
|
echo <<<ENVCONTENT
|
||||||
|
|
||||||
Put the following content into this file:
|
Put the following content into this file:
|
||||||
|
@ -17,6 +17,8 @@
|
|||||||
* - SS_DATABASE_SUFFIX: A suffix to add to the database name.
|
* - SS_DATABASE_SUFFIX: A suffix to add to the database name.
|
||||||
* - SS_DATABASE_PREFIX: A prefix 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_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.
|
* 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
|
* - 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:
|
// For schema enabled drivers:
|
||||||
if(defined('SS_DATABASE_SCHEMA'))
|
if(defined('SS_DATABASE_SCHEMA'))
|
||||||
$databaseConfig["schema"] = 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')) {
|
if(defined('SS_SEND_ALL_EMAILS_TO')) {
|
||||||
|
@ -458,6 +458,8 @@ class Controller extends RequestHandler implements TemplateGlobalProvider {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Redirect to the given URL.
|
* Redirect to the given URL.
|
||||||
|
*
|
||||||
|
* @return SS_HTTPResponse
|
||||||
*/
|
*/
|
||||||
public function redirect($url, $code=302) {
|
public function redirect($url, $code=302) {
|
||||||
if(!$this->response) $this->response = new SS_HTTPResponse();
|
if(!$this->response) $this->response = new SS_HTTPResponse();
|
||||||
@ -473,7 +475,7 @@ class Controller extends RequestHandler implements TemplateGlobalProvider {
|
|||||||
$url = Director::baseURL() . $url;
|
$url = Director::baseURL() . $url;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->response->redirect($url, $code);
|
return $this->response->redirect($url, $code);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -96,6 +96,16 @@ class RequestHandler extends ViewableData {
|
|||||||
* @config
|
* @config
|
||||||
*/
|
*/
|
||||||
private static $allowed_actions = null;
|
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() {
|
public function __construct() {
|
||||||
$this->brokenOnConstruct = false;
|
$this->brokenOnConstruct = false;
|
||||||
@ -430,12 +440,12 @@ class RequestHandler extends ViewableData {
|
|||||||
// If defined as empty array, deny action
|
// If defined as empty array, deny action
|
||||||
$isAllowed = false;
|
$isAllowed = false;
|
||||||
} elseif($allowedActions === null) {
|
} elseif($allowedActions === null) {
|
||||||
// If undefined, allow action
|
// If undefined, allow action based on configuration
|
||||||
$isAllowed = true;
|
$isAllowed = !Config::inst()->get('RequestHandler', 'require_allowed_actions');
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we don't have a match in 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))) {
|
if(!$isDefined && ($action == 'index' || empty($action))) {
|
||||||
$isAllowed = true;
|
$isAllowed = true;
|
||||||
}
|
}
|
||||||
|
@ -86,7 +86,7 @@
|
|||||||
class Session {
|
class Session {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var $timeout Set session timeout
|
* @var $timeout Set session timeout in seconds.
|
||||||
* @config
|
* @config
|
||||||
*/
|
*/
|
||||||
private static $timeout = 0;
|
private static $timeout = 0;
|
||||||
@ -523,9 +523,11 @@ class Session {
|
|||||||
|
|
||||||
if(!session_id() && !headers_sent()) {
|
if(!session_id() && !headers_sent()) {
|
||||||
if($domain) {
|
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 {
|
} 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
|
// 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.
|
// Modify the timeout behaviour so it's the *inactive* time before the session expires.
|
||||||
// By default it's the total session lifetime
|
// By default it's the total session lifetime
|
||||||
if($timeout && !headers_sent()) {
|
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
|
* instead, the last manifest to be added always wins
|
||||||
*/
|
*/
|
||||||
public function pushConfigYamlManifest(SS_ConfigManifest $manifest) {
|
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();
|
$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
|
// @todo: Do anything with these. They're for caching after config.php has executed
|
||||||
$this->collectConfigPHPSettings = true;
|
$this->collectConfigPHPSettings = true;
|
||||||
@ -479,10 +483,13 @@ class Config {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$value = $nothing = null;
|
||||||
|
|
||||||
// Then the manifest values
|
// Then the manifest values
|
||||||
foreach($this->manifests as $manifest) {
|
foreach($this->manifests as $manifest) {
|
||||||
if (isset($manifest[$class][$name])) {
|
$value = $manifest->get($class, $name, $nothing);
|
||||||
self::merge_low_into_high($result, $manifest[$class][$name], $suppress);
|
if ($value !== $nothing) {
|
||||||
|
self::merge_low_into_high($result, $value, $suppress);
|
||||||
if ($result !== null && !is_array($result)) return $result;
|
if ($result !== null && !is_array($result)) return $result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -80,7 +80,7 @@ class PaginatedList extends SS_ListDecorator {
|
|||||||
* @param int $page
|
* @param int $page
|
||||||
*/
|
*/
|
||||||
public function setCurrentPage($page) {
|
public function setCurrentPage($page) {
|
||||||
$this->pageStart = ($page - 1) * $this->pageLength;
|
$this->pageStart = ($page - 1) * $this->getPageLength();
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -91,8 +91,8 @@ class PaginatedList extends SS_ListDecorator {
|
|||||||
*/
|
*/
|
||||||
public function getPageStart() {
|
public function getPageStart() {
|
||||||
if ($this->pageStart === null) {
|
if ($this->pageStart === null) {
|
||||||
if ($this->request && isset($this->request[$this->getVar])) {
|
if ($this->request && isset($this->request[$this->getPaginationGetVar()])) {
|
||||||
$this->pageStart = (int) $this->request[$this->getVar];
|
$this->pageStart = (int) $this->request[$this->getPaginationGetVar()];
|
||||||
} else {
|
} else {
|
||||||
$this->pageStart = 0;
|
$this->pageStart = 0;
|
||||||
}
|
}
|
||||||
@ -181,7 +181,7 @@ class PaginatedList extends SS_ListDecorator {
|
|||||||
if($this->limitItems) {
|
if($this->limitItems) {
|
||||||
$tmptList = clone $this->list;
|
$tmptList = clone $this->list;
|
||||||
return new IteratorIterator(
|
return new IteratorIterator(
|
||||||
$tmptList->limit($this->pageLength, $this->getPageStart())
|
$tmptList->limit($this->getPageLength(), $this->getPageStart())
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
return new IteratorIterator($this->list);
|
return new IteratorIterator($this->list);
|
||||||
@ -223,7 +223,7 @@ class PaginatedList extends SS_ListDecorator {
|
|||||||
for ($i = $start; $i < $end; $i++) {
|
for ($i = $start; $i < $end; $i++) {
|
||||||
$result->push(new ArrayData(array(
|
$result->push(new ArrayData(array(
|
||||||
'PageNum' => $i + 1,
|
'PageNum' => $i + 1,
|
||||||
'Link' => HTTP::setGetVar($this->getVar, $i * $this->pageLength),
|
'Link' => HTTP::setGetVar($this->getPaginationGetVar(), $i * $this->getPageLength()),
|
||||||
'CurrentBool' => $this->CurrentPage() == ($i + 1)
|
'CurrentBool' => $this->CurrentPage() == ($i + 1)
|
||||||
)));
|
)));
|
||||||
}
|
}
|
||||||
@ -292,7 +292,7 @@ class PaginatedList extends SS_ListDecorator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for ($i = 0; $i < $total; $i++) {
|
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;
|
$num = $i + 1;
|
||||||
|
|
||||||
$emptyRange = $num != 1 && $num != $total && (
|
$emptyRange = $num != 1 && $num != $total && (
|
||||||
@ -321,14 +321,14 @@ class PaginatedList extends SS_ListDecorator {
|
|||||||
* @return int
|
* @return int
|
||||||
*/
|
*/
|
||||||
public function CurrentPage() {
|
public function CurrentPage() {
|
||||||
return floor($this->getPageStart() / $this->pageLength) + 1;
|
return floor($this->getPageStart() / $this->getPageLength()) + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return int
|
* @return int
|
||||||
*/
|
*/
|
||||||
public function TotalPages() {
|
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() {
|
public function LastItem() {
|
||||||
if ($start = $this->getPageStart()) {
|
if ($start = $this->getPageStart()) {
|
||||||
return min($start + $this->pageLength, $this->getTotalItems());
|
return min($start + $this->getPageLength(), $this->getTotalItems());
|
||||||
} else {
|
} else {
|
||||||
return min($this->pageLength, $this->getTotalItems());
|
return min($this->getPageLength(), $this->getTotalItems());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -381,7 +381,7 @@ class PaginatedList extends SS_ListDecorator {
|
|||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function FirstLink() {
|
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
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function LastLink() {
|
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() {
|
public function NextLink() {
|
||||||
if ($this->NotLastPage()) {
|
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() {
|
public function PrevLink() {
|
||||||
if ($this->NotFirstPage()) {
|
if ($this->NotFirstPage()) {
|
||||||
return HTTP::setGetVar($this->getVar, $this->getPageStart() - $this->pageLength);
|
return HTTP::setGetVar($this->getPaginationGetVar(), $this->getPageStart() - $this->getPageLength());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -9,7 +9,16 @@
|
|||||||
*/
|
*/
|
||||||
class SS_ConfigManifest {
|
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
|
* All the values needed to be collected to determine the correct combination of fragements for
|
||||||
* the current environment.
|
* the current environment.
|
||||||
* @var array
|
* @var array
|
||||||
@ -34,6 +43,17 @@ class SS_ConfigManifest {
|
|||||||
*/
|
*/
|
||||||
public $yamlConfig = array();
|
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
|
* 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
|
* 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 ) {
|
public function __construct($base, $includeTests = false, $forceRegen = false ) {
|
||||||
$this->base = $base;
|
$this->base = $base;
|
||||||
|
$this->key = sha1($base).'_';
|
||||||
|
$this->includeTests = $includeTests;
|
||||||
|
|
||||||
// Get the Zend Cache to load/store cache into
|
// Get the Zend Cache to load/store cache into
|
||||||
$this->cache = SS_Cache::factory('SS_Configuration', 'Core', array(
|
$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
|
// Unless we're forcing regen, try loading from cache
|
||||||
if (!$forceRegen) {
|
if (!$forceRegen) {
|
||||||
// The PHP config sources are always needed
|
// 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
|
// Get the variant key spec
|
||||||
$this->variantKeySpec = $this->cache->load('variant_key_spec');
|
$this->variantKeySpec = $this->cache->load($this->key.'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();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we don't have a config yet, we need to do a full regen to get it
|
// If we don't have a variantKeySpec (because we're forcing regen, or it just wasn't in the cache), generate it
|
||||||
if (!$this->yamlConfig) {
|
if (!$this->variantKeySpec) {
|
||||||
$this->regenerate($includeTests);
|
$this->regenerate($includeTests);
|
||||||
$this->buildYamlConfigVariant();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 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) {
|
foreach ($this->phpConfigSources as $config) {
|
||||||
require_once $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();
|
$this->buildVariantKeySpec();
|
||||||
|
|
||||||
if ($cache) {
|
if ($cache) {
|
||||||
$this->cache->save($this->phpConfigSources, 'php_config_sources');
|
$this->cache->save($this->phpConfigSources, $this->key.'php_config_sources');
|
||||||
$this->cache->save($this->yamlConfigFragments, 'yaml_config_fragments');
|
$this->cache->save($this->yamlConfigFragments, $this->key.'yaml_config_fragments');
|
||||||
$this->cache->save($this->variantKeySpec, 'variant_key_spec');
|
$this->cache->save($this->variantKeySpec, $this->key.'variant_key_spec');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -395,10 +440,17 @@ class SS_ConfigManifest {
|
|||||||
$matchingFragments = array();
|
$matchingFragments = array();
|
||||||
|
|
||||||
foreach ($this->yamlConfigFragments as $i => $fragment) {
|
foreach ($this->yamlConfigFragments as $i => $fragment) {
|
||||||
$failsonly = isset($fragment['only']) && !$this->matchesPrefilterVariantRules($fragment['only']);
|
$matches = true;
|
||||||
$matchesexcept = isset($fragment['except']) && $this->matchesPrefilterVariantRules($fragment['except']);
|
|
||||||
|
|
||||||
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;
|
$this->yamlConfigFragments = $matchingFragments;
|
||||||
@ -413,22 +465,26 @@ class SS_ConfigManifest {
|
|||||||
* which values means accept or reject a fragment
|
* which values means accept or reject a fragment
|
||||||
*/
|
*/
|
||||||
public function matchesPrefilterVariantRules($rules) {
|
public function matchesPrefilterVariantRules($rules) {
|
||||||
|
$matches = "undefined"; // Needs to be truthy, but not true
|
||||||
|
|
||||||
foreach ($rules as $k => $v) {
|
foreach ($rules as $k => $v) {
|
||||||
switch (strtolower($k)) {
|
switch (strtolower($k)) {
|
||||||
case 'classexists':
|
case 'classexists':
|
||||||
if (!ClassInfo::exists($v)) return false;
|
$matches = $matches && ClassInfo::exists($v);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'moduleexists':
|
case 'moduleexists':
|
||||||
if (!$this->moduleExists($v)) return false;
|
$matches = $matches && $this->moduleExists($v);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
// NOP
|
// 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
|
* Calculates which yaml config fragments are applicable in this variant, and merge those all together into
|
||||||
* the $this->yamlConfig propperty
|
* 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) {
|
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->yamlConfig = array();
|
||||||
|
$this->yamlConfigVariantKey = $this->variantKey();
|
||||||
|
|
||||||
foreach ($this->yamlConfigFragments as $i => $fragment) {
|
foreach ($this->yamlConfigFragments as $i => $fragment) {
|
||||||
$failsonly = isset($fragment['only']) && !$this->matchesVariantRules($fragment['only']);
|
$matches = true;
|
||||||
$matchesexcept = isset($fragment['except']) && $this->matchesVariantRules($fragment['except']);
|
|
||||||
|
|
||||||
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) {
|
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
|
* Returns false if the non-prefilterable parts of the rule aren't met, and true if they are
|
||||||
*/
|
*/
|
||||||
public function matchesVariantRules($rules) {
|
public function matchesVariantRules($rules) {
|
||||||
|
$matches = "undefined"; // Needs to be truthy, but not true
|
||||||
|
|
||||||
foreach ($rules as $k => $v) {
|
foreach ($rules as $k => $v) {
|
||||||
switch (strtolower($k)) {
|
switch (strtolower($k)) {
|
||||||
case 'classexists':
|
case 'classexists':
|
||||||
@ -510,13 +601,13 @@ class SS_ConfigManifest {
|
|||||||
case 'environment':
|
case 'environment':
|
||||||
switch (strtolower($v)) {
|
switch (strtolower($v)) {
|
||||||
case 'live':
|
case 'live':
|
||||||
if (!Director::isLive()) return false;
|
$matches = $matches && Director::isLive();
|
||||||
break;
|
break;
|
||||||
case 'test':
|
case 'test':
|
||||||
if (!Director::isTest()) return false;
|
$matches = $matches && Director::isTest();
|
||||||
break;
|
break;
|
||||||
case 'dev':
|
case 'dev':
|
||||||
if (!Director::isDev()) return false;
|
$matches = $matches && Director::isDev();
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
user_error('Unknown environment '.$v.' in config fragment', E_USER_ERROR);
|
user_error('Unknown environment '.$v.' in config fragment', E_USER_ERROR);
|
||||||
@ -524,21 +615,25 @@ class SS_ConfigManifest {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case 'envvarset':
|
case 'envvarset':
|
||||||
if (isset($_ENV[$k])) break;
|
$matches = $matches && isset($_ENV[$v]);
|
||||||
return false;
|
break;
|
||||||
|
|
||||||
case 'constantdefined':
|
case 'constantdefined':
|
||||||
if (defined($k)) break;
|
$matches = $matches && defined($v);
|
||||||
return false;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
if (isset($_ENV[$k]) && $_ENV[$k] == $v) break;
|
$matches = $matches && (
|
||||||
if (defined($k) && constant($k) == $v) break;
|
(isset($_ENV[$k]) && $_ENV[$k] == $v) ||
|
||||||
return false;
|
(defined($k) && constant($k) == $v)
|
||||||
|
);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($matches === false) return $matches;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return $matches;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -78,7 +78,8 @@ class SS_ConfigStaticManifest {
|
|||||||
$static = $this->statics[$class][$name];
|
$static = $this->statics[$class][$name];
|
||||||
|
|
||||||
if ($static['access'] != T_PRIVATE) {
|
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
|
// Don't warn more than once per static
|
||||||
$this->statics[$class][$name]['access'] = T_PRIVATE;
|
$this->statics[$class][$name]['access'] = T_PRIVATE;
|
||||||
}
|
}
|
||||||
@ -211,13 +212,23 @@ class SS_ConfigStaticManifest_Parser {
|
|||||||
$class = $next[1];
|
$class = $next[1];
|
||||||
}
|
}
|
||||||
else if($type == T_NAMESPACE) {
|
else if($type == T_NAMESPACE) {
|
||||||
$next = $this->next();
|
$namespace = '';
|
||||||
|
while(true) {
|
||||||
|
$next = $this->next();
|
||||||
|
|
||||||
if($next[0] != T_STRING) {
|
if($next == ';') {
|
||||||
user_error("Couldn\'t parse {$this->path} when building config static manifest", E_USER_ERROR);
|
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){
|
else if($type == '{' || $type == T_CURLY_OPEN || $type == T_DOLLAR_OPEN_CURLY_BRACES){
|
||||||
$depth += 1;
|
$depth += 1;
|
||||||
@ -288,7 +299,8 @@ class SS_ConfigStaticManifest_Parser {
|
|||||||
$depth -= 1;
|
$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) {
|
if($type == T_WHITESPACE) {
|
||||||
$value .= ' ';
|
$value .= ' ';
|
||||||
}
|
}
|
||||||
|
@ -43,4 +43,4 @@ Used in side panels and action tabs
|
|||||||
.ss-uploadfield .ss-uploadfield-addfile.borderTop { border-top: 1px solid #b3b3b3; }
|
.ss-uploadfield .ss-uploadfield-addfile.borderTop { border-top: 1px solid #b3b3b3; }
|
||||||
|
|
||||||
.ss-upload .clear { clear: both; }
|
.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->setSuite($suite);
|
||||||
$phpunitwrapper->setCoverageStatus($coverage);
|
$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();
|
$phpunitwrapper->runTests();
|
||||||
|
|
||||||
// get results of the PhpUnitWrapper class
|
// get results of the PhpUnitWrapper class
|
||||||
|
@ -35,6 +35,7 @@
|
|||||||
* Optional integration with ImageMagick as a new image manipulation backend
|
* Optional integration with ImageMagick as a new image manipulation backend
|
||||||
* Support for PHP 5.4's built-in webserver
|
* Support for PHP 5.4's built-in webserver
|
||||||
* Support for [Composer](http://getcomposer.org) dependency manager (also works with 3.0)
|
* 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
|
## 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
|
In order to make controller access checks more consistent and easier to
|
||||||
understand, the routing will require definition of `$allowed_actions`
|
understand, the routing will require definition of `$allowed_actions`
|
||||||
on your own `Controller` subclasses if they contain any actions
|
on your own `Controller` subclasses if they contain any actions accessible through URLs.
|
||||||
accessible through URLs, or any forms.
|
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
class MyController extends Controller {
|
class MyController extends Controller {
|
||||||
@ -228,8 +228,13 @@ accessible through URLs, or any forms.
|
|||||||
public function myaction($request) {}
|
public function myaction($request) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
Please review all rules governing allowed actions in the
|
You can overwrite the default behaviour on undefined `$allowed_actions` to allow all actions,
|
||||||
["controller" topic](/topics/controller).
|
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`
|
### 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.
|
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
|
:::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.
|
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)
|
![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
|
### 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.
|
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
|
||||||
|
|
||||||
GridField is SilverStripe's implementation of data grids. Its main purpose is to display tabular data
|
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.
|
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
|
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.
|
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
|
:::ss
|
||||||
<% if $MyDinner=="quiche" %>
|
<% if $MyDinner=="quiche" %>
|
||||||
Real men don't eat quiche
|
Real men don't eat quiche
|
||||||
<% else_if $MyDinner=$YourDinner %>
|
<% else_if $MyDinner==$YourDinner %>
|
||||||
We both have good taste
|
We both have good taste
|
||||||
<% else %>
|
<% else %>
|
||||||
Can I have some of your chips?
|
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.
|
current environment.
|
||||||
|
|
||||||
To achieve this you add a key to the related header section, either "Only" when the value section should be included
|
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
|
only when all the rules contained match, or "Except" when the value section should be included except when all of the
|
||||||
contained match.
|
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.
|
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'
|
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
|
||||||
|
|
||||||
The values section of YAML configuration files is quite simple - it is simply a nested key / value pair structure
|
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:
|
There's a couple of rules guiding these checks:
|
||||||
|
|
||||||
* Each class is only responsible for access control on the methods it defines
|
* 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 an empty array or undefined, only the `index` action is allowed
|
||||||
* If `$allowed_actions` is undefined, all public methods on the specific class are allowed
|
|
||||||
(not recommended)
|
|
||||||
* Access checks on parent classes need to be overwritten via the Config API
|
* Access checks on parent classes need to be overwritten via the Config API
|
||||||
* Only public methods can be made accessible
|
* 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
|
* 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.
|
* `$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.
|
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
|
### Through the action
|
||||||
|
|
||||||
@ -173,21 +173,6 @@ through `/fastfood/drivethrough/` to use the same order function.
|
|||||||
'drivethrough/$Action/$ID/$Name' => 'order'
|
'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
|
## URL Patterns
|
||||||
|
|
||||||
The `[api:RequestHandler]` class will parse all rules you specify against the
|
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.
|
* 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.
|
* 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)
|
## Cross-Site Request Forgery (CSRF)
|
||||||
|
|
||||||
|
@ -56,16 +56,16 @@ The `phpunit` binary should be used from the root directory of your website.
|
|||||||
|
|
||||||
# Runs all tests defined in phpunit.xml
|
# Runs all tests defined in phpunit.xml
|
||||||
phpunit
|
phpunit
|
||||||
|
|
||||||
# Run all tests of a specific module
|
# Run all tests of a specific module
|
||||||
phpunit framework/tests/
|
phpunit framework/tests/
|
||||||
|
|
||||||
# Run specific tests within a specific module
|
# Run specific tests within a specific module
|
||||||
phpunit framework/tests/filesystem
|
phpunit framework/tests/filesystem
|
||||||
|
|
||||||
# Run a specific test
|
# Run a specific test
|
||||||
phpunit framework/tests/filesystem/FolderTest.php
|
phpunit framework/tests/filesystem/FolderTest.php
|
||||||
|
|
||||||
# Run tests with optional `$_GET` parameters (you need an empty second argument)
|
# Run tests with optional `$_GET` parameters (you need an empty second argument)
|
||||||
phpunit framework/tests '' flush=all
|
phpunit framework/tests '' flush=all
|
||||||
|
|
||||||
@ -81,16 +81,16 @@ particularly around formatting test output.
|
|||||||
|
|
||||||
# Run all tests
|
# Run all tests
|
||||||
sake dev/tests/all
|
sake dev/tests/all
|
||||||
|
|
||||||
# Run all tests of a specific module (comma-separated)
|
# Run all tests of a specific module (comma-separated)
|
||||||
sake dev/tests/module/framework,cms
|
sake dev/tests/module/framework,cms
|
||||||
|
|
||||||
# Run specific tests (comma-separated)
|
# Run specific tests (comma-separated)
|
||||||
sake dev/tests/FolderTest,OtherTest
|
sake dev/tests/FolderTest,OtherTest
|
||||||
|
|
||||||
# Run tests with optional `$_GET` parameters
|
# Run tests with optional `$_GET` parameters
|
||||||
sake dev/tests/all flush=all
|
sake dev/tests/all flush=all
|
||||||
|
|
||||||
# Skip some tests
|
# Skip some tests
|
||||||
sake dev/tests/all SkipTests=MySkippedTest
|
sake dev/tests/all SkipTests=MySkippedTest
|
||||||
|
|
||||||
@ -187,4 +187,4 @@ understand the problem space and discover suitable APIs for performing specific
|
|||||||
**Behavior Driven Development (BDD):** An extension of the test-driven programming style, where tests are used primarily
|
**Behavior Driven Development (BDD):** An extension of the test-driven programming style, where tests are used primarily
|
||||||
for describing the specification of how code should perform. In practice, there's little or no technical difference - it
|
for describing the specification of how code should perform. In practice, there's little or no technical difference - it
|
||||||
all comes down to language. In BDD, the usual terminology is changed to reflect this change of focus, so *Specification*
|
all comes down to language. In BDD, the usual terminology is changed to reflect this change of focus, so *Specification*
|
||||||
is used in place of *Test Case*, and *should* is used in place of *expect* and *assert*.
|
is used in place of *Test Case*, and *should* is used in place of *expect* and *assert*.
|
||||||
|
@ -247,7 +247,10 @@ class ConfirmedPasswordField extends FormField {
|
|||||||
*
|
*
|
||||||
* @return ConfirmedPasswordField
|
* @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(is_array($value)) {
|
||||||
if($value['_Password'] || (!$value['_Password'] && !$this->canBeEmpty)) {
|
if($value['_Password'] || (!$value['_Password'] && !$this->canBeEmpty)) {
|
||||||
$this->value = $value['_Password'];
|
$this->value = $value['_Password'];
|
||||||
|
@ -149,6 +149,12 @@ class Form extends RequestHandler {
|
|||||||
*/
|
*/
|
||||||
protected $attributes = array();
|
protected $attributes = array();
|
||||||
|
|
||||||
|
private static $allowed_actions = array(
|
||||||
|
'handleField',
|
||||||
|
'httpSubmission',
|
||||||
|
'forTemplate',
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var FormTemplateHelper
|
* @var FormTemplateHelper
|
||||||
*/
|
*/
|
||||||
@ -223,7 +229,7 @@ class Form extends RequestHandler {
|
|||||||
'GET ' => 'httpSubmission',
|
'GET ' => 'httpSubmission',
|
||||||
'HEAD ' => 'httpSubmission',
|
'HEAD ' => 'httpSubmission',
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set up current form errors in session to
|
* Set up current form errors in session to
|
||||||
* the current form if appropriate.
|
* the current form if appropriate.
|
||||||
@ -313,7 +319,7 @@ class Form extends RequestHandler {
|
|||||||
Form::set_current_action($funcName);
|
Form::set_current_action($funcName);
|
||||||
$this->setButtonClicked($funcName);
|
$this->setButtonClicked($funcName);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Permission checks (first on controller, then falling back to form)
|
// Permission checks (first on controller, then falling back to form)
|
||||||
if(
|
if(
|
||||||
// Ensure that the action is actually a button or method on the form,
|
// Ensure that the action is actually a button or method on the form,
|
||||||
@ -375,6 +381,19 @@ class Form extends RequestHandler {
|
|||||||
return $this->httpError(404);
|
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
|
* Returns the appropriate response up the controller chain
|
||||||
* if {@link validate()} fails (which is checked prior to executing any form actions).
|
* 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)) {
|
if($field = $this->checkFieldsForAction($field->FieldList(), $funcName)) {
|
||||||
return $field;
|
return $field;
|
||||||
}
|
}
|
||||||
} elseif ($field->hasMethod($funcName)) {
|
} elseif ($field->hasMethod($funcName) && $field->checkAccessAction($funcName)) {
|
||||||
return $field;
|
return $field;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -14,6 +14,18 @@ class HtmlEditorField extends TextareaField {
|
|||||||
*/
|
*/
|
||||||
private static $use_gzip = true;
|
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;
|
protected $rows = 30;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -105,7 +117,12 @@ class HtmlEditorField extends TextareaField {
|
|||||||
$linkedFiles = array();
|
$linkedFiles = array();
|
||||||
|
|
||||||
$htmlValue = Injector::inst()->create('HTMLValue', $this->value);
|
$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')) {
|
if(class_exists('SiteTree')) {
|
||||||
// Populate link tracking for internal links & links to asset files.
|
// Populate link tracking for internal links & links to asset files.
|
||||||
if($links = $htmlValue->getElementsByTagName('a')) foreach($links as $link) {
|
if($links = $htmlValue->getElementsByTagName('a')) foreach($links as $link) {
|
||||||
@ -428,8 +445,6 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
|||||||
$computerUploadField->removeExtraClass('ss-uploadfield');
|
$computerUploadField->removeExtraClass('ss-uploadfield');
|
||||||
$computerUploadField->setTemplate('HtmlEditorField_UploadField');
|
$computerUploadField->setTemplate('HtmlEditorField_UploadField');
|
||||||
$computerUploadField->setFolderName(Config::inst()->get('Upload', 'uploads_folder'));
|
$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(
|
$tabSet = new TabSet(
|
||||||
"MediaFormInsertMediaTabs",
|
"MediaFormInsertMediaTabs",
|
||||||
@ -622,10 +637,10 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
|||||||
'CSSClass',
|
'CSSClass',
|
||||||
_t('HtmlEditorField.CSSCLASS', 'Alignment / style'),
|
_t('HtmlEditorField.CSSCLASS', 'Alignment / style'),
|
||||||
array(
|
array(
|
||||||
'left' => _t('HtmlEditorField.CSSCLASSLEFT', 'On the left, with text wrapping around.'),
|
|
||||||
'leftAlone' => _t('HtmlEditorField.CSSCLASSLEFTALONE', 'On the left, on its own.'),
|
'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.'),
|
'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')
|
)->addExtraClass('last')
|
||||||
);
|
);
|
||||||
@ -636,12 +651,12 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
|||||||
TextField::create(
|
TextField::create(
|
||||||
'Width',
|
'Width',
|
||||||
_t('HtmlEditorField.IMAGEWIDTHPX', 'Width'),
|
_t('HtmlEditorField.IMAGEWIDTHPX', 'Width'),
|
||||||
$file->Width
|
$file->InsertWidth
|
||||||
)->setMaxLength(5),
|
)->setMaxLength(5),
|
||||||
TextField::create(
|
TextField::create(
|
||||||
'Height',
|
'Height',
|
||||||
_t('HtmlEditorField.IMAGEHEIGHTPX', 'Height'),
|
_t('HtmlEditorField.IMAGEHEIGHTPX', 'Height'),
|
||||||
$file->Height
|
$file->InsertHeight
|
||||||
)->setMaxLength(5)
|
)->setMaxLength(5)
|
||||||
)->addExtraClass('dimensions last')
|
)->addExtraClass('dimensions last')
|
||||||
);
|
);
|
||||||
@ -661,7 +676,7 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
|||||||
), 'CaptionText');
|
), 'CaptionText');
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->extend('updateFieldsForImage', $fields, $url, $file);
|
$this->extend('updateFieldsForOembed', $fields, $url, $file);
|
||||||
|
|
||||||
return $fields;
|
return $fields;
|
||||||
}
|
}
|
||||||
@ -746,10 +761,10 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
|||||||
'CSSClass',
|
'CSSClass',
|
||||||
_t('HtmlEditorField.CSSCLASS', 'Alignment / style'),
|
_t('HtmlEditorField.CSSCLASS', 'Alignment / style'),
|
||||||
array(
|
array(
|
||||||
'left' => _t('HtmlEditorField.CSSCLASSLEFT', 'On the left, with text wrapping around.'),
|
|
||||||
'leftAlone' => _t('HtmlEditorField.CSSCLASSLEFTALONE', 'On the left, on its own.'),
|
'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.'),
|
'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')
|
)->addExtraClass('last')
|
||||||
);
|
);
|
||||||
@ -759,12 +774,12 @@ class HtmlEditorField_Toolbar extends RequestHandler {
|
|||||||
TextField::create(
|
TextField::create(
|
||||||
'Width',
|
'Width',
|
||||||
_t('HtmlEditorField.IMAGEWIDTHPX', 'Width'),
|
_t('HtmlEditorField.IMAGEWIDTHPX', 'Width'),
|
||||||
$file->Width
|
$file->InsertWidth
|
||||||
)->setMaxLength(5),
|
)->setMaxLength(5),
|
||||||
TextField::create(
|
TextField::create(
|
||||||
'Height',
|
'Height',
|
||||||
" x " . _t('HtmlEditorField.IMAGEHEIGHTPX', 'Height'),
|
" x " . _t('HtmlEditorField.IMAGEHEIGHTPX', 'Height'),
|
||||||
$file->Height
|
$file->InsertHeight
|
||||||
)->setMaxLength(5)
|
)->setMaxLength(5)
|
||||||
)->addExtraClass('dimensions last')
|
)->addExtraClass('dimensions last')
|
||||||
);
|
);
|
||||||
@ -910,6 +925,29 @@ class HtmlEditorField_Embed extends HtmlEditorField_File {
|
|||||||
return $this->oembed->Height ?: 100;
|
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() {
|
public function getPreview() {
|
||||||
if(isset($this->oembed->thumbnail_url)) {
|
if(isset($this->oembed->thumbnail_url)) {
|
||||||
return sprintf('<img src="%s" />', $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;
|
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() {
|
public function getPreview() {
|
||||||
return ($this->file) ? $this->file->CMSThumbnail() : sprintf('<img src="%s" />', $this->url);
|
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',
|
'' => 'index',
|
||||||
);
|
);
|
||||||
|
|
||||||
|
private static $allowed_actions = array(
|
||||||
|
'delete',
|
||||||
|
'edit',
|
||||||
|
'EditForm',
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param UploadFIeld $parent
|
* @param UploadFIeld $parent
|
||||||
* @param int $item
|
* @param int $item
|
||||||
@ -1484,6 +1490,10 @@ class UploadField_SelectHandler extends RequestHandler {
|
|||||||
'' => 'index',
|
'' => 'index',
|
||||||
);
|
);
|
||||||
|
|
||||||
|
private static $allowed_actions = array(
|
||||||
|
'Form'
|
||||||
|
);
|
||||||
|
|
||||||
public function __construct($parent, $folderName = null) {
|
public function __construct($parent, $folderName = null) {
|
||||||
$this->parent = $parent;
|
$this->parent = $parent;
|
||||||
$this->folderName = $folderName;
|
$this->folderName = $folderName;
|
||||||
|
@ -194,6 +194,12 @@ class GridFieldDetailForm implements GridField_URLHandler {
|
|||||||
* @subpackage fields-gridfield
|
* @subpackage fields-gridfield
|
||||||
*/
|
*/
|
||||||
class GridFieldDetailForm_ItemRequest extends RequestHandler {
|
class GridFieldDetailForm_ItemRequest extends RequestHandler {
|
||||||
|
|
||||||
|
private static $allowed_actions = array(
|
||||||
|
'edit',
|
||||||
|
'view',
|
||||||
|
'ItemEditForm'
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -382,7 +382,7 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
|||||||
success: function(html) {
|
success: function(html) {
|
||||||
dialog.html(html);
|
dialog.html(html);
|
||||||
dialog.getForm().setElement(self);
|
dialog.getForm().setElement(self);
|
||||||
dialog.trigger('dialogopen');
|
dialog.trigger('ssdialogopen');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -452,7 +452,7 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
|||||||
},
|
},
|
||||||
|
|
||||||
fromDialog: {
|
fromDialog: {
|
||||||
ondialogopen: function(){
|
onssdialogopen: function(){
|
||||||
var ed = this.getEditor();
|
var ed = this.getEditor();
|
||||||
ed.onopen();
|
ed.onopen();
|
||||||
|
|
||||||
@ -467,7 +467,7 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
|||||||
this.redraw();
|
this.redraw();
|
||||||
},
|
},
|
||||||
|
|
||||||
ondialogclose: function(){
|
onssdialogclose: function(){
|
||||||
var ed = this.getEditor();
|
var ed = this.getEditor();
|
||||||
ed.onclose();
|
ed.onclose();
|
||||||
|
|
||||||
@ -826,7 +826,7 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
|||||||
});
|
});
|
||||||
|
|
||||||
ed.repaint();
|
ed.repaint();
|
||||||
})
|
});
|
||||||
|
|
||||||
this.getDialog().close();
|
this.getDialog().close();
|
||||||
return false;
|
return false;
|
||||||
@ -944,8 +944,9 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
|||||||
var uploadedFiles = $('.ss-uploadfield-files', this).children('.ss-uploadfield-item');
|
var uploadedFiles = $('.ss-uploadfield-files', this).children('.ss-uploadfield-item');
|
||||||
uploadedFiles.each(function(){
|
uploadedFiles.each(function(){
|
||||||
var uploadedID = $(this).data('fileid');
|
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
|
//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);
|
form.showFileView(uploadedID);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -1271,12 +1272,6 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
|||||||
this._super();
|
this._super();
|
||||||
|
|
||||||
this.setOrigVal(parseInt(this.val(), 10));
|
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() {
|
onunmatch: function() {
|
||||||
this._super();
|
this._super();
|
||||||
|
@ -4,8 +4,22 @@
|
|||||||
* On resize of any close the open treedropdownfields
|
* On resize of any close the open treedropdownfields
|
||||||
* as we'll need to redo with widths
|
* as we'll need to redo with widths
|
||||||
*/
|
*/
|
||||||
$(window).resize(function() {
|
var windowWidth, windowHeight;
|
||||||
$('.TreeDropdownField').closePanel();
|
$(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 = {
|
var strings = {
|
||||||
|
@ -64,13 +64,16 @@
|
|||||||
.addClass('ui-state-warning-text');
|
.addClass('ui-state-warning-text');
|
||||||
data.context.find('.ss-uploadfield-item-progress').hide();
|
data.context.find('.ss-uploadfield-item-progress').hide();
|
||||||
data.context.find('.ss-uploadfield-item-overwrite').show();
|
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-progress').show();
|
||||||
data.context.find('.ss-uploadfield-item-overwrite').hide();
|
data.context.find('.ss-uploadfield-item-overwrite').hide();
|
||||||
data.context.find('.ss-uploadfield-item-status')
|
data.context.find('.ss-uploadfield-item-status')
|
||||||
.removeClass('ui-state-warning-text');
|
.removeClass('ui-state-warning-text');
|
||||||
//upload only if the "overwrite" button is clicked
|
//upload only if the "overwrite" button is clicked
|
||||||
$.blueimpUI.fileupload.prototype._onSend.call(that, e, data);
|
$.blueimpUI.fileupload.prototype._onSend.call(that, e, data);
|
||||||
|
|
||||||
|
e.preventDefault(); // Avoid a form submit
|
||||||
|
return false;
|
||||||
});
|
});
|
||||||
} else { //regular file upload
|
} else { //regular file upload
|
||||||
return $.blueimpUI.fileupload.prototype._onSend.call(that, e, data);
|
return $.blueimpUI.fileupload.prototype._onSend.call(that, e, data);
|
||||||
@ -319,12 +322,14 @@
|
|||||||
$('div.ss-upload .ss-uploadfield-startall').entwine({
|
$('div.ss-upload .ss-uploadfield-startall').entwine({
|
||||||
onclick: function(e) {
|
onclick: function(e) {
|
||||||
this.closest('.ss-upload').find('.ss-uploadfield-item-start button').click();
|
this.closest('.ss-upload').find('.ss-uploadfield-item-start button').click();
|
||||||
|
e.preventDefault(); // Avoid a form submit
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
$('div.ss-upload .ss-uploadfield-item-cancelfailed').entwine({
|
$('div.ss-upload .ss-uploadfield-item-cancelfailed').entwine({
|
||||||
onclick: function(e) {
|
onclick: function(e) {
|
||||||
this.closest('.ss-uploadfield-item').remove();
|
this.closest('.ss-uploadfield-item').remove();
|
||||||
|
e.preventDefault(); // Avoid a form submit
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -349,6 +354,7 @@
|
|||||||
fileupload._trigger('destroy', e, {context: item});
|
fileupload._trigger('destroy', e, {context: item});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
e.preventDefault(); // Avoid a form submit
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -371,6 +377,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
e.preventDefault(); // Avoid a form submit
|
e.preventDefault(); // Avoid a form submit
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
$( 'div.ss-upload:not(.disabled):not(.readonly) .ss-uploadfield-item-edit').entwine({
|
$( 'div.ss-upload:not(.disabled):not(.readonly) .ss-uploadfield-item-edit').entwine({
|
||||||
@ -403,6 +410,7 @@
|
|||||||
editform.toggleEditForm();
|
editform.toggleEditForm();
|
||||||
}
|
}
|
||||||
e.preventDefault(); // Avoid a form submit
|
e.preventDefault(); // Avoid a form submit
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -489,8 +497,9 @@
|
|||||||
});
|
});
|
||||||
$('div.ss-upload .ss-uploadfield-fromfiles').entwine({
|
$('div.ss-upload .ss-uploadfield-fromfiles').entwine({
|
||||||
onclick: function(e) {
|
onclick: function(e) {
|
||||||
e.preventDefault();
|
|
||||||
this.getUploadField().openSelectDialog(this.closest('.ss-uploadfield-item'));
|
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_custom_database_fields = array();
|
||||||
protected static $_cache_field_labels = 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.
|
* Non-static relationship cache, indexed by component name.
|
||||||
*/
|
*/
|
||||||
@ -223,6 +231,8 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
}
|
}
|
||||||
|
|
||||||
return array_merge (
|
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 (
|
array (
|
||||||
'ClassName' => self::$classname_spec_cache[$class],
|
'ClassName' => self::$classname_spec_cache[$class],
|
||||||
'Created' => 'SS_Datetime',
|
'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.
|
* 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
|
* 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
|
* @param string $fieldName Limit the output to a specific field name
|
||||||
* @return array The database fields
|
* @return array The database fields
|
||||||
*/
|
*/
|
||||||
public function db($fieldName = null) {
|
public function db($fieldName = null) {
|
||||||
|
if ($fieldName && array_key_exists($fieldName, self::$fixed_fields)) {
|
||||||
|
return self::$fixed_fields[$fieldName];
|
||||||
|
}
|
||||||
|
|
||||||
$classes = ClassInfo::ancestry($this);
|
$classes = ClassInfo::ancestry($this);
|
||||||
$good = false;
|
$good = false;
|
||||||
$items = array();
|
$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;
|
return $items;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2387,15 +2406,7 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
* @return boolean
|
* @return boolean
|
||||||
*/
|
*/
|
||||||
public function hasDatabaseField($field) {
|
public function hasDatabaseField($field) {
|
||||||
// Add base fields which are not defined in static $db
|
if(isset(self::$fixed_fields[$field])) return true;
|
||||||
static $fixedFields = array(
|
|
||||||
'ID' => 'Int',
|
|
||||||
'ClassName' => 'Enum',
|
|
||||||
'LastEdited' => 'SS_Datetime',
|
|
||||||
'Created' => 'SS_Datetime',
|
|
||||||
);
|
|
||||||
|
|
||||||
if(isset($fixedFields[$field])) return true;
|
|
||||||
|
|
||||||
return array_key_exists($field, $this->inheritedDatabaseFields());
|
return array_key_exists($field, $this->inheritedDatabaseFields());
|
||||||
}
|
}
|
||||||
@ -2658,9 +2669,14 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
// Special case for ID field
|
// Special case for ID field
|
||||||
} else if($fieldName == 'ID') {
|
} else if($fieldName == 'ID') {
|
||||||
return new PrimaryKey($fieldName, $this);
|
return new PrimaryKey($fieldName, $this);
|
||||||
|
|
||||||
// General casting information for items in $db or $casting
|
// Special case for ClassName
|
||||||
} else if($helper = $this->castingHelper($fieldName)) {
|
} 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 = Object::create_from_string($helper, $fieldName);
|
||||||
$obj->setValue($this->$fieldName, $this->record, false);
|
$obj->setValue($this->$fieldName, $this->record, false);
|
||||||
return $obj;
|
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))) {
|
} else if(preg_match('/ID$/', $fieldName) && $this->has_one(substr($fieldName,0,-2))) {
|
||||||
$val = $this->$fieldName;
|
$val = $this->$fieldName;
|
||||||
return DBField::create_field('ForeignKey', $val, $fieldName, $this);
|
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;
|
top: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
border: solid #000;
|
|
||||||
border-width: 0 0 100px 200px;
|
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
filter: alpha(opacity=0);
|
filter: alpha(opacity=0);
|
||||||
-o-transform: translate(250px, -50px) scale(1);
|
transform: translate(-300px, 0) scale(4);
|
||||||
-moz-transform: translate(-300px, 0) scale(4);
|
font-size: 23px;
|
||||||
direction: ltr;
|
direction: ltr;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
height: 30px;
|
||||||
|
line-height: 30px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -300,4 +300,4 @@ abstract class SearchFilter extends Object {
|
|||||||
else return null;
|
else return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -346,7 +346,9 @@ class Security extends Controller {
|
|||||||
$member = Member::currentUser();
|
$member = Member::currentUser();
|
||||||
if($member) $member->logOut();
|
if($member) $member->logOut();
|
||||||
|
|
||||||
if($redirect) $this->redirectBack();
|
if($redirect && (!$this->response || !$this->response->isFinished())) {
|
||||||
|
$this->redirectBack();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -363,13 +363,13 @@ class CmsUiContext extends BehatContext
|
|||||||
}
|
}
|
||||||
|
|
||||||
assertNotNull($container, 'Chosen.js field container not found');
|
assertNotNull($container, 'Chosen.js field container not found');
|
||||||
|
|
||||||
// Click on newly expanded list element, indirectly setting the dropdown value
|
// Click on newly expanded list element, indirectly setting the dropdown value
|
||||||
$linkEl = $container->find('xpath', './/a[./@href]');
|
$linkEl = $container->find('xpath', './/a[./@href]');
|
||||||
assertNotNull($linkEl, 'Chosen.js link element not found');
|
assertNotNull($linkEl, 'Chosen.js link element not found');
|
||||||
$this->getSession()->wait(100); // wait for dropdown overlay to appear
|
$this->getSession()->wait(100); // wait for dropdown overlay to appear
|
||||||
$linkEl->click();
|
$linkEl->click();
|
||||||
|
|
||||||
if(in_array('treedropdown', explode(' ', $container->getAttribute('class')))) {
|
if(in_array('treedropdown', explode(' ', $container->getAttribute('class')))) {
|
||||||
// wait for ajax dropdown to load
|
// wait for ajax dropdown to load
|
||||||
$this->getSession()->wait(
|
$this->getSession()->wait(
|
||||||
@ -422,7 +422,7 @@ class CmsUiContext extends BehatContext
|
|||||||
) {
|
) {
|
||||||
if($container->isVisible() && in_array($class, explode(' ', $container->getAttribute('class')))) {
|
if($container->isVisible() && in_array($class, explode(' ', $container->getAttribute('class')))) {
|
||||||
return $container;
|
return $container;
|
||||||
}
|
}
|
||||||
$container = $container->getParent();
|
$container = $container->getParent();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
@assets
|
||||||
Feature: Insert an image into a page
|
Feature: Insert an image into a page
|
||||||
As a cms author
|
As a cms author
|
||||||
I want to insert an image into a page
|
I want to insert an image into a page
|
||||||
|
@ -5,7 +5,7 @@ class ControllerTest extends FunctionalTest {
|
|||||||
protected static $fixture_file = 'ControllerTest.yml';
|
protected static $fixture_file = 'ControllerTest.yml';
|
||||||
|
|
||||||
protected $autoFollowRedirection = false;
|
protected $autoFollowRedirection = false;
|
||||||
|
|
||||||
protected $requiredExtensions = array(
|
protected $requiredExtensions = array(
|
||||||
'ControllerTest_AccessBaseController' => array(
|
'ControllerTest_AccessBaseController' => array(
|
||||||
'ControllerTest_AccessBaseControllerExtension'
|
'ControllerTest_AccessBaseControllerExtension'
|
||||||
@ -44,7 +44,7 @@ class ControllerTest extends FunctionalTest {
|
|||||||
|
|
||||||
public function testAllowedActions() {
|
public function testAllowedActions() {
|
||||||
$adminUser = $this->objFromFixture('Member', 'admin');
|
$adminUser = $this->objFromFixture('Member', 'admin');
|
||||||
|
|
||||||
$response = $this->get("ControllerTest_UnsecuredController/");
|
$response = $this->get("ControllerTest_UnsecuredController/");
|
||||||
$this->assertEquals(200, $response->getStatusCode(),
|
$this->assertEquals(200, $response->getStatusCode(),
|
||||||
'Access granted on index action without $allowed_actions on defining controller, ' .
|
'Access granted on index action without $allowed_actions on defining controller, ' .
|
||||||
@ -53,16 +53,32 @@ class ControllerTest extends FunctionalTest {
|
|||||||
|
|
||||||
$response = $this->get("ControllerTest_UnsecuredController/index");
|
$response = $this->get("ControllerTest_UnsecuredController/index");
|
||||||
$this->assertEquals(200, $response->getStatusCode(),
|
$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'
|
'when called with an action in the URL'
|
||||||
);
|
);
|
||||||
|
|
||||||
$response = $this->get("ControllerTest_UnsecuredController/method1");
|
Config::inst()->update('RequestHandler', 'require_allowed_actions', false);
|
||||||
|
$response = $this->get("ControllerTest_UnsecuredController/index");
|
||||||
$this->assertEquals(200, $response->getStatusCode(),
|
$this->assertEquals(200, $response->getStatusCode(),
|
||||||
'Access granted on action without $allowed_actions on defining controller, ' .
|
'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'
|
'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, and explicitly allowed through config'
|
||||||
|
);
|
||||||
|
Config::inst()->update('RequestHandler', 'require_allowed_actions', true);
|
||||||
|
|
||||||
$response = $this->get("ControllerTest_AccessBaseController/");
|
$response = $this->get("ControllerTest_AccessBaseController/");
|
||||||
$this->assertEquals(200, $response->getStatusCode(),
|
$this->assertEquals(200, $response->getStatusCode(),
|
||||||
'Access granted on index with empty $allowed_actions on defining controller, ' .
|
'Access granted on index with empty $allowed_actions on defining controller, ' .
|
||||||
@ -110,6 +126,12 @@ class ControllerTest extends FunctionalTest {
|
|||||||
'if action is not a method but rather a template discovered by naming convention'
|
'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);
|
$this->session()->inst_set('loggedInAs', $adminUser->ID);
|
||||||
$response = $this->get("ControllerTest_AccessSecuredController/templateaction");
|
$response = $this->get("ControllerTest_AccessSecuredController/templateaction");
|
||||||
$this->assertEquals(200, $response->getStatusCode(),
|
$this->assertEquals(200, $response->getStatusCode(),
|
||||||
@ -147,25 +169,25 @@ class ControllerTest extends FunctionalTest {
|
|||||||
"Access granted to method defined in allowed_actions on extension, " .
|
"Access granted to method defined in allowed_actions on extension, " .
|
||||||
"where method is also defined on extension"
|
"where method is also defined on extension"
|
||||||
);
|
);
|
||||||
|
|
||||||
$response = $this->get('ControllerTest_AccessSecuredController/extensionmethod1');
|
$response = $this->get('ControllerTest_AccessSecuredController/extensionmethod1');
|
||||||
$this->assertEquals(200, $response->getStatusCode(),
|
$this->assertEquals(200, $response->getStatusCode(),
|
||||||
"Access granted to method defined in allowed_actions on extension, " .
|
"Access granted to method defined in allowed_actions on extension, " .
|
||||||
"where method is also defined on extension, even when called in a subclass"
|
"where method is also defined on extension, even when called in a subclass"
|
||||||
);
|
);
|
||||||
|
|
||||||
$response = $this->get('ControllerTest_AccessBaseController/extensionmethod2');
|
$response = $this->get('ControllerTest_AccessBaseController/extensionmethod2');
|
||||||
$this->assertEquals(404, $response->getStatusCode(),
|
$this->assertEquals(404, $response->getStatusCode(),
|
||||||
"Access denied to method not defined in allowed_actions on extension, " .
|
"Access denied to method not defined in allowed_actions on extension, " .
|
||||||
"where method is also defined on extension"
|
"where method is also defined on extension"
|
||||||
);
|
);
|
||||||
|
|
||||||
$response = $this->get('ControllerTest_IndexSecuredController/');
|
$response = $this->get('ControllerTest_IndexSecuredController/');
|
||||||
$this->assertEquals(403, $response->getStatusCode(),
|
$this->assertEquals(403, $response->getStatusCode(),
|
||||||
"Access denied when index action is limited through allowed_actions, " .
|
"Access denied when index action is limited through allowed_actions, " .
|
||||||
"and doesn't satisfy checks, and action is empty"
|
"and doesn't satisfy checks, and action is empty"
|
||||||
);
|
);
|
||||||
|
|
||||||
$response = $this->get('ControllerTest_IndexSecuredController/index');
|
$response = $this->get('ControllerTest_IndexSecuredController/index');
|
||||||
$this->assertEquals(403, $response->getStatusCode(),
|
$this->assertEquals(403, $response->getStatusCode(),
|
||||||
"Access denied when index action is limited through allowed_actions, " .
|
"Access denied when index action is limited through allowed_actions, " .
|
||||||
@ -174,13 +196,13 @@ class ControllerTest extends FunctionalTest {
|
|||||||
|
|
||||||
$this->session()->inst_set('loggedInAs', $adminUser->ID);
|
$this->session()->inst_set('loggedInAs', $adminUser->ID);
|
||||||
$response = $this->get('ControllerTest_IndexSecuredController/');
|
$response = $this->get('ControllerTest_IndexSecuredController/');
|
||||||
$this->assertEquals(200, $response->getStatusCode(),
|
$this->assertEquals(200, $response->getStatusCode(),
|
||||||
"Access granted when index action is limited through allowed_actions, " .
|
"Access granted when index action is limited through allowed_actions, " .
|
||||||
"and does satisfy checks"
|
"and does satisfy checks"
|
||||||
);
|
);
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
$this->session()->inst_set('loggedInAs', null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @expectedException PHPUnit_Framework_Error
|
* @expectedException PHPUnit_Framework_Error
|
||||||
* @expectedExceptionMessage Wildcards (*) are no longer valid
|
* @expectedExceptionMessage Wildcards (*) are no longer valid
|
||||||
@ -358,7 +380,7 @@ class ControllerTest extends FunctionalTest {
|
|||||||
class ControllerTest_Controller extends Controller implements TestOnly {
|
class ControllerTest_Controller extends Controller implements TestOnly {
|
||||||
|
|
||||||
public $Content = "default content";
|
public $Content = "default content";
|
||||||
|
|
||||||
private static $allowed_actions = array(
|
private static $allowed_actions = array(
|
||||||
'methodaction',
|
'methodaction',
|
||||||
'stringaction',
|
'stringaction',
|
||||||
@ -385,13 +407,13 @@ class ControllerTest_UnsecuredController extends Controller implements TestOnly
|
|||||||
|
|
||||||
// Not defined, allow access to all
|
// Not defined, allow access to all
|
||||||
// static $allowed_actions = array();
|
// static $allowed_actions = array();
|
||||||
|
|
||||||
// Granted for all
|
// Granted for all
|
||||||
public function method1() {}
|
public function method1() {}
|
||||||
|
|
||||||
// Granted for all
|
// Granted for all
|
||||||
public function method2() {}
|
public function method2() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
class ControllerTest_AccessBaseController extends Controller implements TestOnly {
|
class ControllerTest_AccessBaseController extends Controller implements TestOnly {
|
||||||
|
|
||||||
@ -402,7 +424,7 @@ class ControllerTest_AccessBaseController extends Controller implements TestOnly
|
|||||||
|
|
||||||
// Denied for all
|
// Denied for all
|
||||||
public function method2() {}
|
public function method2() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
class ControllerTest_AccessSecuredController extends ControllerTest_AccessBaseController implements TestOnly {
|
class ControllerTest_AccessSecuredController extends ControllerTest_AccessBaseController implements TestOnly {
|
||||||
|
|
||||||
@ -414,7 +436,7 @@ class ControllerTest_AccessSecuredController extends ControllerTest_AccessBaseCo
|
|||||||
);
|
);
|
||||||
|
|
||||||
public function method2() {}
|
public function method2() {}
|
||||||
|
|
||||||
public function adminonly() {}
|
public function adminonly() {}
|
||||||
|
|
||||||
protected function protectedmethod() {}
|
protected function protectedmethod() {}
|
||||||
@ -427,18 +449,18 @@ class ControllerTest_AccessWildcardSecuredController extends ControllerTest_Acce
|
|||||||
"*" => "ADMIN", // should throw exception
|
"*" => "ADMIN", // should throw exception
|
||||||
);
|
);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class ControllerTest_IndexSecuredController extends ControllerTest_AccessBaseController implements TestOnly {
|
class ControllerTest_IndexSecuredController extends ControllerTest_AccessBaseController implements TestOnly {
|
||||||
|
|
||||||
private static $allowed_actions = array(
|
private static $allowed_actions = array(
|
||||||
"index" => "ADMIN",
|
"index" => "ADMIN",
|
||||||
);
|
);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class ControllerTest_AccessBaseControllerExtension extends Extension implements TestOnly {
|
class ControllerTest_AccessBaseControllerExtension extends Extension implements TestOnly {
|
||||||
|
|
||||||
private static $allowed_actions = array(
|
private static $allowed_actions = array(
|
||||||
"extensionmethod1" => true, // granted because defined on this class
|
"extensionmethod1" => true, // granted because defined on this class
|
||||||
"method1" => true, // ignored because method not defined on this class
|
"method1" => true, // ignored because method not defined on this class
|
||||||
@ -457,7 +479,7 @@ class ControllerTest_AccessBaseControllerExtension extends Extension implements
|
|||||||
|
|
||||||
public function internalextensionmethod() {}
|
public function internalextensionmethod() {}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class ControllerTest_HasAction extends Controller {
|
class ControllerTest_HasAction extends Controller {
|
||||||
|
|
||||||
|
@ -348,6 +348,13 @@ class DirectorTest extends SapphireTest {
|
|||||||
|
|
||||||
class DirectorTestRequest_Controller extends Controller implements TestOnly {
|
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 returnGetValue($request) { return $_GET['somekey']; }
|
||||||
|
|
||||||
public function returnPostValue($request) { return $_POST['somekey']; }
|
public function returnPostValue($request) { return $_POST['somekey']; }
|
||||||
|
@ -130,10 +130,6 @@ class RequestHandlingTest extends FunctionalTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public function testDisallowedExtendedActions() {
|
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 */
|
/* Actions on an extension are allowed because they specifically provided appropriate allowed_actions items */
|
||||||
$response = Director::test("testGoodBase1/otherExtendedMethod");
|
$response = Director::test("testGoodBase1/otherExtendedMethod");
|
||||||
$this->assertEquals("otherExtendedMethod", $response->getBody());
|
$this->assertEquals("otherExtendedMethod", $response->getBody());
|
||||||
@ -146,9 +142,10 @@ class RequestHandlingTest extends FunctionalTest {
|
|||||||
$response = Director::test("RequestHandlingTest_AllowedController/failoverMethod");
|
$response = Director::test("RequestHandlingTest_AllowedController/failoverMethod");
|
||||||
$this->assertEquals("failoverMethod", $response->getBody());
|
$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");
|
$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 */
|
/* This action has been blocked by an argument to a method */
|
||||||
$response = Director::test('RequestHandlingTest_AllowedController/blockMethod');
|
$response = Director::test('RequestHandlingTest_AllowedController/blockMethod');
|
||||||
@ -421,9 +418,13 @@ class RequestHandlingTest_FormActionController extends Controller {
|
|||||||
* Simple extension for the test controller
|
* Simple extension for the test controller
|
||||||
*/
|
*/
|
||||||
class RequestHandlingTest_ControllerExtension extends Extension {
|
class RequestHandlingTest_ControllerExtension extends Extension {
|
||||||
|
|
||||||
public static $called_error = false;
|
public static $called_error = false;
|
||||||
|
|
||||||
public static $called_404_error = false;
|
public static $called_404_error = false;
|
||||||
|
|
||||||
|
private static $allowed_actions = array('extendedMethod');
|
||||||
|
|
||||||
public function extendedMethod() {
|
public function extendedMethod() {
|
||||||
return "extendedMethod";
|
return "extendedMethod";
|
||||||
}
|
}
|
||||||
@ -455,7 +456,6 @@ class RequestHandlingTest_AllowedController extends Controller implements TestOn
|
|||||||
|
|
||||||
private static $allowed_actions = array(
|
private static $allowed_actions = array(
|
||||||
'failoverMethod', // part of the failover object
|
'failoverMethod', // part of the failover object
|
||||||
'extendedMethod', // part of the RequestHandlingTest_ControllerExtension object
|
|
||||||
'blockMethod' => '->provideAccess(false)',
|
'blockMethod' => '->provideAccess(false)',
|
||||||
'allowMethod' => '->provideAccess',
|
'allowMethod' => '->provideAccess',
|
||||||
);
|
);
|
||||||
@ -537,6 +537,8 @@ class RequestHandlingTest_Form extends Form {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class RequestHandlingTest_ControllerFormWithAllowedActions extends Controller implements TestOnly {
|
class RequestHandlingTest_ControllerFormWithAllowedActions extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
public function Form() {
|
public function Form() {
|
||||||
return new RequestHandlingTest_FormWithAllowedActions(
|
return new RequestHandlingTest_FormWithAllowedActions(
|
||||||
@ -544,8 +546,7 @@ class RequestHandlingTest_ControllerFormWithAllowedActions extends Controller im
|
|||||||
'Form',
|
'Form',
|
||||||
new FieldList(),
|
new FieldList(),
|
||||||
new FieldList(
|
new FieldList(
|
||||||
new FormAction('allowedformaction'),
|
new FormAction('allowedformaction')
|
||||||
new FormAction('disallowedformaction') // disallowed through $allowed_actions in form
|
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -555,7 +556,6 @@ class RequestHandlingTest_FormWithAllowedActions extends Form {
|
|||||||
|
|
||||||
private static $allowed_actions = array(
|
private static $allowed_actions = array(
|
||||||
'allowedformaction' => 1,
|
'allowedformaction' => 1,
|
||||||
'httpSubmission' => 1, // TODO This should be an exception on the parent class
|
|
||||||
);
|
);
|
||||||
|
|
||||||
public function allowedformaction() {
|
public function allowedformaction() {
|
||||||
@ -603,6 +603,9 @@ class RequestHandlingTest_FormField extends FormField {
|
|||||||
* Form field for the test
|
* Form field for the test
|
||||||
*/
|
*/
|
||||||
class RequestHandlingTest_SubclassedFormField extends RequestHandlingTest_FormField {
|
class RequestHandlingTest_SubclassedFormField extends RequestHandlingTest_FormField {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('customSomething');
|
||||||
|
|
||||||
// We have some url_handlers defined that override RequestHandlingTest_FormField handlers.
|
// We have some url_handlers defined that override RequestHandlingTest_FormField handlers.
|
||||||
// We will confirm that the url_handlers inherit.
|
// We will confirm that the url_handlers inherit.
|
||||||
private static $url_handlers = array(
|
private static $url_handlers = array(
|
||||||
@ -620,6 +623,8 @@ class RequestHandlingTest_SubclassedFormField extends RequestHandlingTest_FormFi
|
|||||||
* Controller for the test
|
* Controller for the test
|
||||||
*/
|
*/
|
||||||
class RequestHandlingFieldTest_Controller extends Controller implements TestOnly {
|
class RequestHandlingFieldTest_Controller extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('TestForm');
|
||||||
|
|
||||||
public function TestForm() {
|
public function TestForm() {
|
||||||
return new Form($this, "TestForm", new FieldList(
|
return new Form($this, "TestForm", new FieldList(
|
||||||
@ -642,4 +647,8 @@ class RequestHandlingTest_HandlingField extends FormField {
|
|||||||
public function actionOnField() {
|
public function actionOnField() {
|
||||||
return "Test method on $this->name";
|
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 {
|
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() {
|
public function testRelativeOrder() {
|
||||||
$accessor = new ConfigManifestTest_ConfigManifestAccess(BASE_PATH, true, false);
|
$accessor = new ConfigManifestTest_ConfigManifestAccess(BASE_PATH, true, false);
|
||||||
|
|
||||||
@ -88,4 +256,4 @@ class ConfigManifestTest extends SapphireTest {
|
|||||||
), 'after');
|
), 'after');
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -170,7 +170,8 @@ DOC;
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$parser = new SS_ConfigStaticManifest_Parser(__DIR__ . '/ConfigStaticManifestTest/ConfigStaticManifestTestMyObject.php');
|
$parser = new SS_ConfigStaticManifest_Parser(__DIR__ .
|
||||||
|
'/ConfigStaticManifestTest/ConfigStaticManifestTestMyObject.php');
|
||||||
$parser->parse();
|
$parser->parse();
|
||||||
|
|
||||||
$statics = $parser->getStatics();
|
$statics = $parser->getStatics();
|
||||||
@ -182,4 +183,19 @@ DOC;
|
|||||||
|
|
||||||
$this->assertEquals($expectedValue, $statics['ConfigStaticManifestTestMyObject']['db']['value']);
|
$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());
|
$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() {
|
public function testSetShowOnClick() {
|
||||||
//hide by default and display show/hide toggle button
|
//hide by default and display show/hide toggle button
|
||||||
$field = new ConfirmedPasswordField('Test', 'Testing', 'valueA', null, true);
|
$field = new ConfirmedPasswordField('Test', 'Testing', 'valueA', null, true);
|
||||||
|
@ -73,6 +73,8 @@ class EmailFieldTest_Validator extends Validator {
|
|||||||
|
|
||||||
class EmailFieldTest_Controller extends Controller implements TestOnly {
|
class EmailFieldTest_Controller extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
private static $url_handlers = array(
|
private static $url_handlers = array(
|
||||||
'$Action//$ID/$OtherID' => "handleAction",
|
'$Action//$ID/$OtherID' => "handleAction",
|
||||||
);
|
);
|
||||||
|
@ -479,6 +479,9 @@ class FormTest_Team extends DataObject implements TestOnly {
|
|||||||
* @subpackage tests
|
* @subpackage tests
|
||||||
*/
|
*/
|
||||||
class FormTest_Controller extends Controller implements TestOnly {
|
class FormTest_Controller extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
private static $url_handlers = array(
|
private static $url_handlers = array(
|
||||||
'$Action//$ID/$OtherID' => "handleAction",
|
'$Action//$ID/$OtherID' => "handleAction",
|
||||||
);
|
);
|
||||||
@ -528,6 +531,9 @@ class FormTest_Controller extends Controller implements TestOnly {
|
|||||||
* @subpackage tests
|
* @subpackage tests
|
||||||
*/
|
*/
|
||||||
class FormTest_ControllerWithSecurityToken extends Controller implements TestOnly {
|
class FormTest_ControllerWithSecurityToken extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
private static $url_handlers = array(
|
private static $url_handlers = array(
|
||||||
'$Action//$ID/$OtherID' => "handleAction",
|
'$Action//$ID/$OtherID' => "handleAction",
|
||||||
);
|
);
|
||||||
@ -562,6 +568,9 @@ class FormTest_ControllerWithSecurityToken extends Controller implements TestOnl
|
|||||||
}
|
}
|
||||||
|
|
||||||
class FormTest_ControllerWithStrictPostCheck extends Controller implements TestOnly {
|
class FormTest_ControllerWithStrictPostCheck extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
protected $template = 'BlankPage';
|
protected $template = 'BlankPage';
|
||||||
|
|
||||||
public function Link($action = null) {
|
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 {
|
class GridFieldAddExistingAutocompleterTest_Controller extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
protected $template = 'BlankPage';
|
protected $template = 'BlankPage';
|
||||||
|
|
||||||
public function Form() {
|
public function Form() {
|
||||||
|
@ -282,6 +282,7 @@ class GridFieldDetailFormTest_PeopleGroup extends DataObject implements TestOnly
|
|||||||
}
|
}
|
||||||
|
|
||||||
class GridFieldDetailFormTest_Category extends DataObject implements TestOnly {
|
class GridFieldDetailFormTest_Category extends DataObject implements TestOnly {
|
||||||
|
|
||||||
private static $db = array(
|
private static $db = array(
|
||||||
'Name' => 'Varchar'
|
'Name' => 'Varchar'
|
||||||
);
|
);
|
||||||
@ -306,6 +307,9 @@ class GridFieldDetailFormTest_Category extends DataObject implements TestOnly {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class GridFieldDetailFormTest_Controller extends Controller implements TestOnly {
|
class GridFieldDetailFormTest_Controller extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
protected $template = 'BlankPage';
|
protected $template = 'BlankPage';
|
||||||
|
|
||||||
public function Form() {
|
public function Form() {
|
||||||
@ -326,6 +330,9 @@ class GridFieldDetailFormTest_Controller extends Controller implements TestOnly
|
|||||||
}
|
}
|
||||||
|
|
||||||
class GridFieldDetailFormTest_GroupController extends Controller implements TestOnly {
|
class GridFieldDetailFormTest_GroupController extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
protected $template = 'BlankPage';
|
protected $template = 'BlankPage';
|
||||||
|
|
||||||
public function Form() {
|
public function Form() {
|
||||||
@ -339,6 +346,9 @@ class GridFieldDetailFormTest_GroupController extends Controller implements Test
|
|||||||
}
|
}
|
||||||
|
|
||||||
class GridFieldDetailFormTest_CategoryController extends Controller implements TestOnly {
|
class GridFieldDetailFormTest_CategoryController extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
protected $template = 'BlankPage';
|
protected $template = 'BlankPage';
|
||||||
|
|
||||||
public function Form() {
|
public function Form() {
|
||||||
|
@ -30,6 +30,9 @@ class GridField_URLHandlerTest extends FunctionalTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class GridField_URLHandlerTest_Controller extends Controller implements TestOnly {
|
class GridField_URLHandlerTest_Controller extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form');
|
||||||
|
|
||||||
public function Link() {
|
public function Link() {
|
||||||
return get_class($this) ."/";
|
return get_class($this) ."/";
|
||||||
}
|
}
|
||||||
@ -51,6 +54,9 @@ class GridField_URLHandlerTest_Controller extends Controller implements TestOnly
|
|||||||
* Test URLHandler with a nested request handler
|
* Test URLHandler with a nested request handler
|
||||||
*/
|
*/
|
||||||
class GridField_URLHandlerTest_Component extends RequestHandler implements GridField_URLHandler {
|
class GridField_URLHandlerTest_Component extends RequestHandler implements GridField_URLHandler {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form', 'showform', 'testpage', 'handleItem');
|
||||||
|
|
||||||
protected $gridField;
|
protected $gridField;
|
||||||
|
|
||||||
public function getURLHandlers($gridField) {
|
public function getURLHandlers($gridField) {
|
||||||
@ -96,8 +102,13 @@ class GridField_URLHandlerTest_Component extends RequestHandler implements GridF
|
|||||||
}
|
}
|
||||||
|
|
||||||
class GridField_URLHandlerTest_Component_ItemRequest extends RequestHandler {
|
class GridField_URLHandlerTest_Component_ItemRequest extends RequestHandler {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('Form', 'showform', 'testpage');
|
||||||
|
|
||||||
protected $gridField;
|
protected $gridField;
|
||||||
|
|
||||||
protected $link;
|
protected $link;
|
||||||
|
|
||||||
protected $id;
|
protected $id;
|
||||||
|
|
||||||
public function __construct($gridField, $id, $link) {
|
public function __construct($gridField, $id, $link) {
|
||||||
|
@ -21,7 +21,47 @@ class DataListTest extends SapphireTest {
|
|||||||
'DataObjectTest_TeamComment',
|
'DataObjectTest_TeamComment',
|
||||||
'DataObjectTest\NamespacedClass',
|
'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(){
|
public function testSubtract(){
|
||||||
$comment1 = $this->objFromFixture('DataObjectTest_TeamComment', 'comment1');
|
$comment1 = $this->objFromFixture('DataObjectTest_TeamComment', 'comment1');
|
||||||
$subtractList = DataObjectTest_TeamComment::get()->filter('ID', $comment1->ID);
|
$subtractList = DataObjectTest_TeamComment::get()->filter('ID', $comment1->ID);
|
||||||
|
@ -18,7 +18,20 @@ class DataObjectTest extends SapphireTest {
|
|||||||
'DataObjectTest_Player',
|
'DataObjectTest_Player',
|
||||||
'DataObjectTest_TeamComment'
|
'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() {
|
public function testDataIntegrityWhenTwoSubclassesHaveSameField() {
|
||||||
// Save data into DataObjectTest_SubTeam.SubclassDatabaseField
|
// Save data into DataObjectTest_SubTeam.SubclassDatabaseField
|
||||||
$obj = new DataObjectTest_SubTeam();
|
$obj = new DataObjectTest_SubTeam();
|
||||||
|
@ -44,12 +44,12 @@ class GroupTest extends FunctionalTest {
|
|||||||
$form->saveInto($member);
|
$form->saveInto($member);
|
||||||
$updatedGroups = $member->Groups();
|
$updatedGroups = $member->Groups();
|
||||||
|
|
||||||
$this->assertEquals(
|
$this->assertEquals(2, count($updatedGroups->column()),
|
||||||
array($adminGroup->ID, $parentGroup->ID),
|
|
||||||
$updatedGroups->column(),
|
|
||||||
"Adding a toplevel group works"
|
"Adding a toplevel group works"
|
||||||
);
|
);
|
||||||
|
$this->assertContains($adminGroup->ID, $updatedGroups->column('ID'));
|
||||||
|
$this->assertContains($parentGroup->ID, $updatedGroups->column('ID'));
|
||||||
|
|
||||||
// Test unsetting relationship
|
// Test unsetting relationship
|
||||||
$form->loadDataFrom($member);
|
$form->loadDataFrom($member);
|
||||||
$checkboxSetField = $form->Fields()->fieldByName('Groups');
|
$checkboxSetField = $form->Fields()->fieldByName('Groups');
|
||||||
@ -60,11 +60,10 @@ class GroupTest extends FunctionalTest {
|
|||||||
$form->saveInto($member);
|
$form->saveInto($member);
|
||||||
$member->flushCache();
|
$member->flushCache();
|
||||||
$updatedGroups = $member->Groups();
|
$updatedGroups = $member->Groups();
|
||||||
$this->assertEquals(
|
$this->assertEquals(1, count($updatedGroups->column()),
|
||||||
array($adminGroup->ID),
|
|
||||||
$updatedGroups->column(),
|
|
||||||
"Removing a previously added toplevel group works"
|
"Removing a previously added toplevel group works"
|
||||||
);
|
);
|
||||||
|
$this->assertContains($adminGroup->ID, $updatedGroups->column('ID'));
|
||||||
|
|
||||||
// Test adding child group
|
// Test adding child group
|
||||||
|
|
||||||
@ -77,23 +76,21 @@ class GroupTest extends FunctionalTest {
|
|||||||
$orphanGroup->ParentID = 99999;
|
$orphanGroup->ParentID = 99999;
|
||||||
$orphanGroup->write();
|
$orphanGroup->write();
|
||||||
|
|
||||||
$this->assertEquals(
|
$this->assertEquals(1, count($parentGroup->collateAncestorIDs()),
|
||||||
array($parentGroup->ID),
|
|
||||||
$parentGroup->collateAncestorIDs(),
|
|
||||||
'Root node only contains itself'
|
'Root node only contains itself'
|
||||||
);
|
);
|
||||||
|
$this->assertContains($parentGroup->ID, $parentGroup->collateAncestorIDs());
|
||||||
|
|
||||||
$this->assertEquals(
|
$this->assertEquals(2, count($childGroup->collateAncestorIDs()),
|
||||||
array($childGroup->ID, $parentGroup->ID),
|
|
||||||
$childGroup->collateAncestorIDs(),
|
|
||||||
'Contains parent nodes, with child node first'
|
'Contains parent nodes, with child node first'
|
||||||
);
|
);
|
||||||
|
$this->assertContains($parentGroup->ID, $childGroup->collateAncestorIDs());
|
||||||
|
$this->assertContains($childGroup->ID, $childGroup->collateAncestorIDs());
|
||||||
|
|
||||||
$this->assertEquals(
|
$this->assertEquals(1, count($orphanGroup->collateAncestorIDs()),
|
||||||
array($orphanGroup->ID),
|
|
||||||
$orphanGroup->collateAncestorIDs(),
|
|
||||||
'Orphaned nodes dont contain invalid parent IDs'
|
'Orphaned nodes dont contain invalid parent IDs'
|
||||||
);
|
);
|
||||||
|
$this->assertContains($orphanGroup->ID, $orphanGroup->collateAncestorIDs());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDelete() {
|
public function testDelete() {
|
||||||
|
@ -41,8 +41,11 @@ class MemberCsvBulkLoaderTest extends SapphireTest {
|
|||||||
$results = $loader->load($this->getCurrentRelativePath() . '/MemberCsvBulkLoaderTest.csv');
|
$results = $loader->load($this->getCurrentRelativePath() . '/MemberCsvBulkLoaderTest.csv');
|
||||||
|
|
||||||
$created = $results->Created()->toArray();
|
$created = $results->Created()->toArray();
|
||||||
$this->assertEquals($created[0]->Groups()->column('ID'), array($existinggroup->ID));
|
$this->assertEquals(1, count($created[0]->Groups()->column('ID')));
|
||||||
$this->assertEquals($created[1]->Groups()->column('ID'), array($existinggroup->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() {
|
public function testAddToCsvColumnGroupsByCode() {
|
||||||
@ -55,8 +58,12 @@ class MemberCsvBulkLoaderTest extends SapphireTest {
|
|||||||
$this->assertEquals($newgroup->Title, 'newgroup');
|
$this->assertEquals($newgroup->Title, 'newgroup');
|
||||||
|
|
||||||
$created = $results->Created()->toArray();
|
$created = $results->Created()->toArray();
|
||||||
$this->assertEquals($created[0]->Groups()->column('ID'), array($existinggroup->ID));
|
$this->assertEquals(1, count($created[0]->Groups()->column('ID')));
|
||||||
$this->assertEquals($created[1]->Groups()->column('ID'), array($existinggroup->ID, $newgroup->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() {
|
public function testCleartextPasswordsAreHashedWithDefaultAlgo() {
|
||||||
|
@ -439,6 +439,9 @@ class SecurityTest extends FunctionalTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class SecurityTest_SecuredController extends Controller implements TestOnly {
|
class SecurityTest_SecuredController extends Controller implements TestOnly {
|
||||||
|
|
||||||
|
private static $allowed_actions = array('index');
|
||||||
|
|
||||||
public function index() {
|
public function index() {
|
||||||
if(!Permission::check('ADMIN')) return Security::permissionFailure($this);
|
if(!Permission::check('ADMIN')) return Security::permissionFailure($this);
|
||||||
|
|
||||||
|
@ -165,6 +165,18 @@ SS;
|
|||||||
'Permissions template functions result correct result');
|
'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() {
|
public function testLocalFunctionsTakePriorityOverGlobals() {
|
||||||
$data = new ArrayData(array(
|
$data = new ArrayData(array(
|
||||||
'Page' => new SSViewerTest_Object()
|
'Page' => new SSViewerTest_Object()
|
||||||
@ -1062,7 +1074,7 @@ after')
|
|||||||
$origEnv = Config::inst()->get('Director', 'environment_type');
|
$origEnv = Config::inst()->get('Director', 'environment_type');
|
||||||
Config::inst()->update('Director', 'environment_type', 'dev');
|
Config::inst()->update('Director', 'environment_type', 'dev');
|
||||||
Config::inst()->update('SSViewer', 'source_file_comments', true);
|
Config::inst()->update('SSViewer', 'source_file_comments', true);
|
||||||
$f = FRAMEWORK_PATH . '/tests/templates/SSViewerTestComments';
|
$f = FRAMEWORK_PATH . '/tests/templates/SSViewerTestComments';
|
||||||
$templates = array(
|
$templates = array(
|
||||||
array(
|
array(
|
||||||
'name' => 'SSViewerTestCommentsFullSource',
|
'name' => 'SSViewerTestCommentsFullSource',
|
||||||
@ -1078,7 +1090,8 @@ after')
|
|||||||
array(
|
array(
|
||||||
'name' => 'SSViewerTestCommentsFullSourceHTML4Doctype',
|
'name' => 'SSViewerTestCommentsFullSourceHTML4Doctype',
|
||||||
'expected' => ""
|
'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 -->"
|
. "<!-- template $f/SSViewerTestCommentsFullSourceHTML4Doctype.ss -->"
|
||||||
. "<html>"
|
. "<html>"
|
||||||
. "\t<head></head>"
|
. "\t<head></head>"
|
||||||
@ -1197,6 +1210,45 @@ after')
|
|||||||
"tests/forms/RequirementsTest_a.js"
|
"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;
|
public $number = null;
|
||||||
|
|
||||||
|
private static $casting = array(
|
||||||
|
'Link' => 'Text',
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
public function __construct($number = null) {
|
public function __construct($number = null) {
|
||||||
parent::__construct();
|
parent::__construct();
|
||||||
$this->number = $number;
|
$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) {
|
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;
|
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 {
|
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());
|
$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() {
|
public function testRAWVal() {
|
||||||
$data = new ViewableDataTest_Castable();
|
$data = new ViewableDataTest_Castable();
|
||||||
$data->test = 'This & This';
|
$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) {
|
function Lookup__construct(&$res) {
|
||||||
$res['php'] = '$scope';
|
$res['php'] = '$scope->locally()';
|
||||||
$res['LookupSteps'] = array();
|
$res['LookupSteps'] = array();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -157,7 +157,7 @@ class SSTemplateParser extends Parser {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
function Lookup__construct(&$res) {
|
function Lookup__construct(&$res) {
|
||||||
$res['php'] = '$scope';
|
$res['php'] = '$scope->locally()';
|
||||||
$res['LookupSteps'] = array();
|
$res['LookupSteps'] = array();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -49,18 +49,34 @@ class SSViewer_Scope {
|
|||||||
|
|
||||||
public function __construct($item){
|
public function __construct($item){
|
||||||
$this->item = $item;
|
$this->item = $item;
|
||||||
$this->localIndex=0;
|
$this->localIndex = 0;
|
||||||
|
$this->localStack = array();
|
||||||
$this->itemStack[] = array($this->item, null, 0, null, null, 0);
|
$this->itemStack[] = array($this->item, null, 0, null, null, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getItem(){
|
public function getItem(){
|
||||||
return $this->itemIterator ? $this->itemIterator->current() : $this->item;
|
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,
|
list($this->item, $this->itemIterator, $this->itemIteratorTotal, $this->popIndex, $this->upIndex,
|
||||||
$this->currentIndex) = $this->itemStack[$this->localIndex];
|
$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) {
|
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) {
|
if(!is_object($value) && $forceReturnedObject) {
|
||||||
$default = Config::inst()->get('ViewableData', 'default_cast', Config::FIRST_SET);
|
$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;
|
return $value;
|
||||||
|
Loading…
Reference in New Issue
Block a user