From 425881257bf401bd2b92d2b5c049bc63b0e3aefb Mon Sep 17 00:00:00 2001 From: Sean Harvey Date: Wed, 13 Nov 2013 16:46:13 +1300 Subject: [PATCH] BUG Performance improvements of SubmittedFormField queries. When there are a lot of SubmittedForm records the UserDefinedForm page takes a long time to load in the CMS, and oftens exceeds the PHP memory limit well beyond 128M. Previously UserDefinedForm::getCMSFields() would build a list of name => value from all SubmittedFormField records, but it would do this twice, once in getCMSFields() and another time in UserFormsGridFieldFilterHeader. It would also use the full ORM to build this list, when all it needs is a map of the Name and Value columns. This fixes that to build the columns once in getCMSFields() using DB::query() and it'll pass those columns along to UserFormsGridFieldFilterHeader as well so it doesn't do it twice. --- code/UserFormsGridFieldFilterHeader.php | 24 ++++++++++++---------- code/model/UserDefinedForm.php | 27 +++++++++++++++---------- 2 files changed, 29 insertions(+), 22 deletions(-) diff --git a/code/UserFormsGridFieldFilterHeader.php b/code/UserFormsGridFieldFilterHeader.php index 7acfe97..421080b 100644 --- a/code/UserFormsGridFieldFilterHeader.php +++ b/code/UserFormsGridFieldFilterHeader.php @@ -8,7 +8,17 @@ * @package userforms */ class UserFormsGridFieldFilterHeader extends GridFieldFilterHeader { - + + /** + * A map of name => value of columns from all submissions + * @var array + */ + protected $columns; + + public function setColumns($columns) { + $this->columns = $columns; + } + public function handleAction(GridField $gridField, $actionName, $arguments, $data) { if(!$this->checkDataType($gridField->getList())) { return; @@ -36,18 +46,10 @@ class UserFormsGridFieldFilterHeader extends GridFieldFilterHeader { // submitted in this form. $params = $gridField->getForm()->getController()->getURLParams(); - // this is for you SQL server I know you don't see '' as a number - $parentID = (!empty($params['ID'])) ? Convert::raw2sql($params['ID']) : 0; - $formFields = SubmittedFormField::get() - ->where(sprintf("SubmittedForm.ParentID = '%s'", $parentID)) - ->leftJoin('SubmittedForm', 'SubmittedFormField.ParentID = SubmittedForm.ID') - ->sort('Title', 'ASC') - ->map('Name', 'Title'); - // show dropdown of all the fields available from the submitted form fields // that have been saved. Takes the titles from the currently live form. $columnField = new DropdownField('FieldNameFilter', ''); - $columnField->setSource($formFields->toArray()); + $columnField->setSource($this->columns); $columnField->setEmptyString(_t('UserFormsGridFieldFilterHeader.FILTERSUBMISSIONS', 'Filter Submissions..')); $columnField->setHasEmptyDefault(true); $columnField->setValue($selectedField); @@ -143,4 +145,4 @@ class UserFormsGridFieldFilterHeader extends GridFieldFilterHeader { return $dataList; } -} \ No newline at end of file +} diff --git a/code/model/UserDefinedForm.php b/code/model/UserDefinedForm.php index 920bd8b..aceae81 100755 --- a/code/model/UserDefinedForm.php +++ b/code/model/UserDefinedForm.php @@ -62,7 +62,6 @@ class UserDefinedForm extends Page { SiteTree::disableCMSFieldsExtensions(); $fields = parent::getCMSFields(); SiteTree::enableCMSFieldsExtensions(); - // define tabs $fields->findOrMakeTab('Root.FormContent', _t('UserDefinedForm.FORM', 'Form')); $fields->findOrMakeTab('Root.FormOptions', _t('UserDefinedForm.CONFIGURATION', 'Configuration')); @@ -111,6 +110,19 @@ class UserDefinedForm extends Page { $this->Submissions()->sort('Created', 'DESC') ); + // make sure a numeric not a empty string is checked against this int column for SQL server + $parentID = (!empty($this->ID)) ? $this->ID : 0; + + // get a list of all field names and values used for print and export CSV views of the GridField below. + $columnSQL = <<map(); + $config = new GridFieldConfig(); $config->addComponent(new GridFieldToolbarHeader()); $config->addComponent($sort = new GridFieldSortableHeader()); @@ -129,16 +141,9 @@ class UserDefinedForm extends Page { $filter->setThrowExceptionOnBadDataType(false); $pagination->setThrowExceptionOnBadDataType(false); - // make sure a numeric not a empty string is checked against this int column for SQL server - $parentID = (!empty($this->ID)) ? $this->ID : 0; - // attach every column to the print view from - $columns = SubmittedFormField::get() - ->where("\"SubmittedForm\".\"ParentID\" = '$parentID'") - ->leftJoin('SubmittedForm', '"SubmittedFormField"."ParentID" = "SubmittedForm"."ID"') - ->map('Name', 'Title'); - - $columns = $columns->toArray(); - $columns['Created'] = "Created"; + // attach every column to the print view form + $columns['Created'] = 'Created'; + $filter->setColumns($columns); // print configuration $print->setPrintHasHeader(true);