Merge pull request #7729 from colemanw/CRM-17960
[civicrm-core.git] / CRM / Upgrade / Incremental / php / FourSeven.php
CommitLineData
6cc25669
CW
1<?php
2/*
3 +--------------------------------------------------------------------+
4 | CiviCRM version 4.7 |
5 +--------------------------------------------------------------------+
6 | Copyright CiviCRM LLC (c) 2004-2015 |
7 +--------------------------------------------------------------------+
8 | This file is a part of CiviCRM. |
9 | |
10 | CiviCRM is free software; you can copy, modify, and distribute it |
11 | under the terms of the GNU Affero General Public License |
12 | Version 3, 19 November 2007. |
13 | |
14 | CiviCRM is distributed in the hope that it will be useful, but |
15 | WITHOUT ANY WARRANTY; without even the implied warranty of |
16 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. |
17 | See the GNU Affero General Public License for more details. |
18 | |
19 | You should have received a copy of the GNU Affero General Public |
20 | License along with this program; if not, contact CiviCRM LLC |
21 | at info[AT]civicrm[DOT]org. If you have questions about the |
22 | GNU Affero General Public License or the licensing of CiviCRM, |
23 | see the CiviCRM license FAQ at http://civicrm.org/licensing |
24 +--------------------------------------------------------------------+
25 */
26
27/**
bf6a5362 28 * Upgrade logic for 4.7
6cc25669 29 */
bf6a5362 30class CRM_Upgrade_Incremental_php_FourSeven extends CRM_Upgrade_Incremental_Base {
6cc25669 31
f431d51f
J
32 /**
33 * Compute any messages which should be displayed beforeupgrade.
34 *
35 * Note: This function is called iteratively for each upcoming
36 * revision to the database.
37 *
3bdf1f3a 38 * @param string $preUpgradeMessage
f431d51f
J
39 * @param string $rev
40 * a version number, e.g. '4.4.alpha1', '4.4.beta3', '4.4.0'.
41 * @param null $currentVer
f431d51f
J
42 */
43 public function setPreUpgradeMessage(&$preUpgradeMessage, $rev, $currentVer = NULL) {
44 if ($rev == '4.7.alpha1') {
f431d51f
J
45 // CRM-16478 Remove custom fatal error template path option
46 $config = CRM_Core_Config::singleton();
18b3bef6 47 if (!empty($config->fatalErrorTemplate) && $config->fatalErrorTemplate != 'CRM/common/fatal.tpl') {
f431d51f
J
48 $preUpgradeMessage .= '<p>' . ts('The custom fatal error template setting will be removed during the upgrade. You are currently using this custom template: %1 . Following the upgrade you will need to use the standard approach to overriding template files, as described in the documentation.', array(1 => $config->fatalErrorTemplate)) . '</p>';
49 }
f431d51f 50 }
a40fd1ac
CW
51 if ($rev == '4.7.alpha4') {
52 // CRM-17004 Warn of Moneris removal
87a33a95
CW
53 $count = 1;
54 // Query only works in 4.3+
55 if (version_compare($currentVer, "4.3.0") > 0) {
56 $count = CRM_Core_DAO::singleValueQuery("SELECT COUNT(id) FROM civicrm_payment_processor WHERE payment_processor_type_id IN (SELECT id FROM civicrm_payment_processor_type WHERE name = 'Moneris')");
57 }
58 if ($count && !function_exists('moneris_civicrm_managed')) {
a40fd1ac
CW
59 $preUpgradeMessage .= '<p>' . ts('The %1 payment processor is no longer bundled with CiviCRM. After upgrading you will need to install the extension to continue using it.', array(1 => 'Moneris')) . '</p>';
60 }
61 }
f431d51f
J
62 }
63
6cc25669
CW
64 /**
65 * Compute any messages which should be displayed after upgrade.
66 *
67 * @param string $postUpgradeMessage
68 * alterable.
69 * @param string $rev
70 * an intermediate version; note that setPostUpgradeMessage is called repeatedly with different $revs.
6cc25669
CW
71 */
72 public function setPostUpgradeMessage(&$postUpgradeMessage, $rev) {
6dbe2c23
CW
73 if ($rev == '4.7.alpha1') {
74 $config = CRM_Core_Config::singleton();
bf6a5362 75 // FIXME: Performing an upgrade step during postUpgrade message phase is probably bad
6dbe2c23
CW
76 $editor_id = self::updateWysiwyg();
77 $msg = NULL;
78 $ext_href = 'href="' . CRM_Utils_System::url('civicrm/admin/extensions', 'reset=1') . '"';
79 $dsp_href = 'href="' . CRM_Utils_System::url('civicrm/admin/setting/preferences/display', 'reset=1') . '"';
80 $blog_href = 'href="https://civicrm.org/blogs/colemanw/big-changes-wysiwyg-editing-47"';
81 switch ($editor_id) {
82 // TinyMCE
83 case 1:
84 $msg = ts('Your configured editor "TinyMCE" is no longer part of the main CiviCRM download. To continue using it, visit the <a %1>Manage Extensions</a> page to download and install the TinyMCE extension.', array(1 => $ext_href));
85 break;
86
87 // Drupal/Joomla editor
88 case 3:
89 case 4:
90 $msg = ts('CiviCRM no longer integrates with the "%1 Default Editor." Your wysiwyg setting has been reset to the built-in CKEditor. <a %2>Learn more...</a>', array(1 => $config->userFramework, 2 => $blog_href));
91 break;
92 }
93 if ($msg) {
94 $postUpgradeMessage .= '<p>' . $msg . '</p>';
95 }
96 $postUpgradeMessage .= '<p>' . ts('CiviCRM now includes the easy-to-use CKEditor Configurator. To customize the features and display of your wysiwyg editor, visit the <a %1>Display Preferences</a> page. <a %2>Learn more...</a>', array(1 => $dsp_href, 2 => $blog_href)) . '</p>';
dd55005c 97
98 $postUpgradeMessage .= '<br /><br />' . ts('Default version of the following System Workflow Message Templates have been modified: <ul><li>Personal Campaign Pages - Owner Notification</li></ul> If you have modified these templates, please review the new default versions and implement updates as needed to your copies (Administer > Communications > Message Templates > System Workflow Messages).');
f431d51f
J
99
100 $postUpgradeMessage .= '<p>' . ts('The custom fatal error template setting has been removed.') . '</p>';
6dbe2c23 101 }
6cc25669
CW
102 }
103
6cc25669
CW
104 /**
105 * Upgrade function.
106 *
107 * @param string $rev
108 */
109 public function upgrade_4_7_alpha1($rev) {
b604d7ec 110 $this->addTask('Migrate \'on behalf of\' information to module_data', 'migrateOnBehalfOfInfo');
bf6a5362 111 $this->addTask(ts('Upgrade DB to %1: SQL', array(1 => $rev)), 'runSql', $rev);
f806379b 112 $this->addTask(ts('Migrate Settings to %1', array(1 => $rev)), 'migrateSettings', $rev);
b604d7ec 113 $this->addTask('Add Getting Started dashlet', 'addGettingStartedDashlet', $rev);
a40fd1ac
CW
114 }
115
116 /**
117 * Upgrade function.
118 *
119 * @param string $rev
120 */
121 public function upgrade_4_7_alpha4($rev) {
122 $this->addTask(ts('Upgrade DB to %1: SQL', array(1 => $rev)), 'runSql', $rev);
058b8a5e 123 $this->addTask(ts('Remove %1', array(1 => 'Moneris')), 'removePaymentProcessorType', 'Moneris');
b2222b9f 124 $this->addTask('Update Smart Groups', 'fixContactTypeInSmartGroups');
6cc25669
CW
125 }
126
0094ac08
CW
127 /**
128 * Upgrade function.
129 *
130 * @param string $rev
131 */
132 public function upgrade_4_7_beta2($rev) {
133 $this->addTask(ts('Upgrade DB to %1: SQL', array(1 => $rev)), 'runSql', $rev);
134 $this->addTask('Delete unused file', 'deleteVersionCheckCacheFile');
135 }
136
fb1f3850
DRJ
137 /**
138 * Upgrade function.
139 *
140 * @param string $rev
141 */
902e557f 142 public function upgrade_4_7_beta6($rev) {
13599400 143 $this->addTask(ts('Upgrade DB to %1: SQL', array(1 => $rev)), 'runSql', $rev);
fb1f3850 144 $this->addTask('Disable flexible jobs extension', 'disableFlexibleJobsExtension');
20d5377e 145 $this->addTask('Add Index to financial_trxn trxn_id field', 'addIndexFinancialTrxnTrxnID');
fb1f3850
DRJ
146 }
147
8ca47f5c 148 /**
149 * Upgrade function.
150 *
151 * @param string $rev
152 */
153 public function upgrade_4_7_1($rev) {
154 $this->addTask('Add Index to civicrm_contribution creditnote_id field', 'addIndexContributionCreditNoteID');
155 }
156
6cc25669
CW
157 /**
158 * CRM-16354
159 *
6dbe2c23 160 * @return int
6cc25669 161 */
6dbe2c23 162 public static function updateWysiwyg() {
aaffa79f 163 $editorID = Civi::settings()->get('editor_id');
6dbe2c23
CW
164 // Previously a numeric value indicated one of 4 wysiwyg editors shipped in core, and no value indicated 'Textarea'
165 // Now the options are "Textarea", "CKEditor", and the rest have been dropped from core.
166 $newEditor = $editorID ? "CKEditor" : "Textarea";
08ef4ddd 167 Civi::settings()->set('editor_id', $newEditor);
6cc25669 168
6dbe2c23 169 return $editorID;
6cc25669
CW
170 }
171
f806379b
TO
172 /**
173 * Migrate any last remaining options from `civicrm_domain.config_backend` to `civicrm_setting`.
174 * Cleanup setting schema.
175 *
176 * @param CRM_Queue_TaskContext $ctx
177 * @return bool
178 */
179 public function migrateSettings(CRM_Queue_TaskContext $ctx) {
4cc9e637
TO
180 // Tip: If there are problems with adding the new uniqueness index, try inspecting:
181 // SELECT name, domain_id, contact_id, count(*) AS dupes FROM civicrm_setting cs GROUP BY name, domain_id, contact_id HAVING dupes > 1;
182
183 // Nav records are expendable. https://forum.civicrm.org/index.php?topic=36933.0
184 CRM_Core_DAO::executeQuery('DELETE FROM civicrm_setting WHERE contact_id IS NOT NULL AND name = "navigation"');
185
f806379b
TO
186 CRM_Core_DAO::executeQuery('ALTER TABLE civicrm_setting DROP INDEX index_group_name');
187 CRM_Core_DAO::executeQuery('ALTER TABLE civicrm_setting DROP COLUMN group_name');
188 CRM_Core_DAO::executeQuery('ALTER TABLE civicrm_setting ADD UNIQUE INDEX index_domain_contact_name (domain_id, contact_id, name)');
189
190 $domainDao = CRM_Core_DAO::executeQuery('SELECT id, config_backend FROM civicrm_domain');
191 while ($domainDao->fetch()) {
192 $settings = CRM_Upgrade_Incremental_php_FourSeven::convertBackendToSettings($domainDao->id, $domainDao->config_backend);
193 CRM_Core_Error::debug_var('convertBackendToSettings', array(
194 'domainId' => $domainDao->id,
195 'backend' => $domainDao->config_backend,
196 'settings' => $settings,
197 ));
198
199 foreach ($settings as $name => $value) {
200 $rowParams = array(
201 1 => array($domainDao->id, 'Positive'),
202 2 => array($name, 'String'),
203 3 => array(serialize($value), 'String'),
204 );
205 $settingId = CRM_Core_DAO::singleValueQuery(
206 'SELECT id FROM civicrm_setting WHERE domain_id = %1 AND name = %2',
207 $rowParams);
208 if (!$settingId) {
209 CRM_Core_DAO::executeQuery(
210 'INSERT INTO civicrm_setting (domain_id, name, value, is_domain) VALUES (%1,%2,%3,1)',
211 $rowParams);
212 }
213 }
214 }
215
9e726168 216 CRM_Core_DAO::executeQuery('ALTER TABLE civicrm_domain DROP COLUMN config_backend');
f806379b
TO
217
218 return TRUE;
219 }
220
221 /**
222 * Take a config_backend blob and produce an equivalent list of settings.
223 *
224 * @param int $domainId
225 * Domain ID.
226 * @param string $config_backend
227 * Serialized blob.
228 * @return array
229 */
230 public static function convertBackendToSettings($domainId, $config_backend) {
231 if (!$config_backend) {
232 return array();
233 }
234
235 $backend = unserialize($config_backend);
236 if (!$backend) {
237 return array();
238 }
239
240 $mappings = \CRM_Core_Config_MagicMerge::getPropertyMap();
241 $settings = array();
242 foreach ($backend as $propertyName => $propertyValue) {
243 if (isset($mappings[$propertyName][0]) && preg_match('/^setting/', $mappings[$propertyName][0])) {
244 // $mapping format: $propertyName => Array(0 => $type, 1 => $setting|NULL).
245 $settingName = isset($mappings[$propertyName][1]) ? $mappings[$propertyName][1] : $propertyName;
246 $settings[$settingName] = $propertyValue;
247 }
248 }
249
250 return $settings;
251 }
252
0a95c936 253 /**
254 * Add Getting Started dashlet to dashboard
255 *
256 * @param \CRM_Queue_TaskContext $ctx
257 *
258 * @return bool
259 */
260 public function addGettingStartedDashlet(CRM_Queue_TaskContext $ctx) {
e1674273 261 $sql = "SELECT count(*) FROM civicrm_dashboard WHERE name='gettingStarted'";
262 $res = CRM_Core_DAO::singleValueQuery($sql);
263 $domainId = CRM_Core_Config::domainID();
264 if ($res <= 0) {
265 $sql = "INSERT INTO `civicrm_dashboard`
a8b704c5 266 ( `domain_id`, `name`, `label`, `url`, `permission`, `permission_operator`, `column_no`, `is_minimized`, `is_active`, `weight`, `fullscreen_url`, `is_fullscreen`, `is_reserved`) VALUES ( {$domainId}, 'getting-started', 'Getting Started', 'civicrm/dashlet/getting-started?reset=1&snippet=5', 'access CiviCRM', NULL, 0, 0, 1, 0, 'civicrm/dashlet/getting-started?reset=1&snippet=5&context=dashletFullscreen', 1, 1)";
e1674273 267 CRM_Core_DAO::executeQuery($sql);
268 // Add default position for Getting Started Dashlet ( left column)
269 $sql = "INSERT INTO `civicrm_dashboard_contact` (dashboard_id, contact_id, column_no, is_active)
270SELECT (SELECT MAX(id) FROM `civicrm_dashboard`), contact_id, 0, IF (SUM(is_active) > 0, 1, 0)
271FROM `civicrm_dashboard_contact` WHERE 1 GROUP BY contact_id";
272 CRM_Core_DAO::executeQuery($sql);
273 }
0a95c936 274 return TRUE;
e1674273 275 }
4175ee03 276
6b1e1a2c 277 /**
278 * Migrate on-behalf information to uf_join.module_data as on-behalf columns will be dropped
279 * on DB upgrade
280 *
281 * @param CRM_Queue_TaskContext $ctx
282 *
283 * @return bool
284 * TRUE for success
285 */
286 public static function migrateOnBehalfOfInfo(CRM_Queue_TaskContext $ctx) {
d3e92c88 287 $domain = new CRM_Core_DAO_Domain();
288 $domain->find(TRUE);
6b1e1a2c 289
d3e92c88 290 // fetch onBehalf entry in UFJoin table
6b1e1a2c 291 $ufGroupDAO = new CRM_Core_DAO_UFJoin();
292 $ufGroupDAO->module = 'OnBehalf';
293 $ufGroupDAO->find(TRUE);
294
d3e92c88 295 $forOrgColums = array();
296 if ($domain->locales) {
297 $locales = explode(CRM_Core_DAO::VALUE_SEPARATOR, $domain->locales);
298 foreach ($locales as $locale) {
299 $forOrgColums[] = "for_organization_{$locale}";
300 }
301 }
302 else {
303 $forOrgColums[] = "for_organization";
304 }
305
306 $query = "
307 SELECT " . implode(", ", $forOrgColums) . ", uj.id as join_id, uj.uf_group_id as uf_group_id
308 FROM civicrm_contribution_page cp
309 INNER JOIN civicrm_uf_join uj ON uj.entity_id = cp.id AND uj.module = 'OnBehalf'";
310 $dao = CRM_Core_DAO::executeQuery($query, array(), TRUE, NULL, FALSE, FALSE);
6b1e1a2c 311
312 if ($dao->N) {
6b1e1a2c 313 while ($dao->fetch()) {
314 $onBehalfParams['on_behalf'] = array('is_for_organization' => $dao->is_for_organization);
315 if ($domain->locales) {
6b1e1a2c 316 foreach ($locales as $locale) {
317 $for_organization = "for_organization_{$locale}";
318 $onBehalfParams['on_behalf'] += array(
319 $locale => array(
320 'for_organization' => $dao->$for_organization,
321 ),
322 );
323 }
324 }
325 else {
326 $onBehalfParams['on_behalf'] += array(
327 'default' => array(
328 'for_organization' => $dao->for_organization,
329 ),
330 );
331 }
332 $ufJoinParam = array(
333 'id' => $dao->join_id,
334 'module' => 'on_behalf',
d3e92c88 335 'uf_group_id' => $dao->uf_group_id,
6b1e1a2c 336 'module_data' => json_encode($onBehalfParams),
337 );
338 CRM_Core_BAO_UFJoin::create($ufJoinParam);
339 }
340 }
341
342 return TRUE;
343 }
cb804cd9 344
b2222b9f
CW
345 /**
346 * CRM-11782 - Get rid of VALUE_SEPARATOR character in saved search form values
347 *
348 * @param \CRM_Queue_TaskContext $ctx
349 *
350 * @return bool
351 */
352 public function fixContactTypeInSmartGroups(CRM_Queue_TaskContext $ctx) {
353 $sep = CRM_Core_DAO::VALUE_SEPARATOR;
354 $dao = CRM_Core_DAO::executeQuery("SELECT id, form_values FROM civicrm_saved_search WHERE form_values LIKE '%$sep%'");
355 while ($dao->fetch()) {
356 $formValues = unserialize($dao->form_values);
357 if (isset($formValues['contact_type']) && is_array($formValues['contact_type'])) {
358 $newVals = array();
359 foreach ($formValues['contact_type'] as $key => $val) {
360 $newVals[str_replace($sep, '__', $key)] = is_string($val) ? str_replace($sep, '__', $val) : $val;
361 }
362 $formValues['contact_type'] = $newVals;
363 }
364 CRM_Core_DAO::executeQuery("UPDATE civicrm_saved_search SET form_values = %1 WHERE id = {$dao->id}", array(1 => array(serialize($formValues), 'String')));
365 }
366
367 return TRUE;
368 }
369
0094ac08
CW
370 /**
371 * CRM-17637 - Ths file location has been moved; delete the old one
372 *
373 * @param \CRM_Queue_TaskContext $ctx
374 *
375 * @return bool
376 */
377 public function deleteVersionCheckCacheFile(CRM_Queue_TaskContext $ctx) {
378 $config = CRM_Core_Config::singleton();
379 $cacheFile = $config->uploadDir . 'version-info-cache.json';
380 if (file_exists($cacheFile)) {
381 unlink($cacheFile);
382 }
383 return TRUE;
384 }
385
fb1f3850
DRJ
386 /**
387 * CRM-17669 and CRM-17686, make scheduled jobs more flexible, disable the 4.6 extension if installed
388 *
389 * @param \CRM_Queue_TaskContext $ctx
390 *
391 * @return bool
392 */
393 public function disableFlexibleJobsExtension(CRM_Queue_TaskContext $ctx) {
d357221c
DRJ
394 try {
395 civicrm_api3('Extension', 'disable', array('key' => 'com.klangsoft.flexiblejobs'));
396 }
397 catch (CiviCRM_API3_Exception $e) {
398 // just ignore if the extension isn't installed
399 }
fb1f3850
DRJ
400
401 return TRUE;
402 }
403
20d5377e 404 /**
405 * CRM-17752 add index to civicrm_financial_trxn.trxn_id (deliberately non-unique).
406 *
407 * @param \CRM_Queue_TaskContext $ctx
408 *
409 * @return bool
410 */
411 public function addIndexFinancialTrxnTrxnID(CRM_Queue_TaskContext $ctx) {
412 $tables = array('civicrm_financial_trxn' => array('trxn_id'));
413 CRM_Core_BAO_SchemaHandler::createIndexes($tables);
414 return TRUE;
415 }
416
8ca47f5c 417 /**
418 * CRM-17882 Add index to civicrm_contribution.credit_note_id.
419 *
420 * @param \CRM_Queue_TaskContext $ctx
421 *
422 * @return bool
423 */
424 public function addIndexContributionCreditNoteID(CRM_Queue_TaskContext $ctx) {
425 $tables = array('civicrm_contribution' => array('creditnote_id'));
426 CRM_Core_BAO_SchemaHandler::createIndexes($tables);
427 return TRUE;
428 }
429
6cc25669 430}