PageRenderTime 55ms CodeModel.GetById 25ms RepoModel.GetById 1ms app.codeStats 0ms

/wp-content/plugins/w3-total-cache/ConfigCompiler.php

https://bitbucket.org/dimonkok/hal
PHP | 451 lines | 302 code | 88 blank | 61 comment | 67 complexity | 19163d15effe5ad34a3b0e5c5281d473 MD5 | raw file
Possible License(s): GPL-2.0, Apache-2.0, BSD-3-Clause, MIT, GPL-3.0
  1. <?php
  2. namespace W3TC;
  3. class ConfigCompiler {
  4. private $_blog_id;
  5. private $_preview;
  6. private $_data;
  7. private $_keys;
  8. /**
  9. * Returns true is key is not modifiable anymore
  10. * Placed here to limit class loading
  11. *
  12. * @return bool
  13. */
  14. static public function child_key_sealed( $key, $master_data, $child_data ) {
  15. if ( isset( $master_data['common.force_master'] ) &&
  16. (boolean)$master_data['common.force_master'] )
  17. return true;
  18. // affects rules which are global, not possible to overload
  19. if ( $key == 'pgcache.engine' &&
  20. ( $master_data['pgcache.engine'] == 'file_generic' ) )
  21. return true;
  22. if ( in_array( $key, array(
  23. 'minify.rewrite',
  24. 'browsercache.rewrite',
  25. 'version' ) ) )
  26. return true;
  27. include W3TC_DIR . '/ConfigKeys.php';
  28. // key which marks overloads is always editable
  29. $overloads_postfix = '.configuration_overloaded';
  30. // extension settings sealing
  31. // e.g. array('newrelic' , '...') is controlled
  32. // by 'newrelic.configuration_overloaded']
  33. $block_key = ( is_array( $key ) ? $key[0] : $key );
  34. if ( isset( $child_data[$block_key] ) &&
  35. isset( $child_data[$block_key . $overloads_postfix] ) &&
  36. (boolean)$child_data[$block_key . $overloads_postfix] )
  37. return false;
  38. if ( !is_array( $key ) ) {
  39. if ( substr( $key, strlen( $key ) - strlen( $overloads_postfix ),
  40. strlen( $overloads_postfix ) ) == $overloads_postfix )
  41. return false;
  42. // default sealing
  43. foreach ( $overloading_keys_scope as $i ) {
  44. $overloading_key = $i['key'];
  45. // check if this key is allowed by overloading-mark key
  46. if ( substr( $key, 0, strlen( $i['prefix'] ) ) == $i['prefix'] ) {
  47. if ( !isset( $child_data[$overloading_key] ) )
  48. return true;
  49. if ( (boolean)$child_data[$overloading_key] )
  50. return false;
  51. }
  52. }
  53. }
  54. return true;
  55. }
  56. /**
  57. * Reads config from file and returns it's content as array (or null)
  58. * Stored in this class to limit class loading
  59. */
  60. static private function util_array_from_file_legacy_v1( $filename ) {
  61. if ( file_exists( $filename ) && is_readable( $filename ) ) {
  62. // including file directly instead of read+eval causes constant
  63. // problems with APC, ZendCache, and WSOD in a case of
  64. // broken config file
  65. $content = @file_get_contents( $filename );
  66. $config = @eval( substr( $content, 5 ) );
  67. if ( is_array( $config ) )
  68. return $config;
  69. }
  70. return null;
  71. }
  72. /**
  73. * Reads config from file and returns it's content as array (or null)
  74. * Stored in this class to limit class loading
  75. */
  76. static private function util_array_from_file_legacy_v2( $filename ) {
  77. if ( file_exists( $filename ) && is_readable( $filename ) ) {
  78. // including file directly instead of read+eval causes constant
  79. // problems with APC, ZendCache, and WSOD in a case of
  80. // broken config file
  81. $content = @file_get_contents( $filename );
  82. $config = @json_decode( $content, true );
  83. if ( is_array( $config ) )
  84. return $config;
  85. }
  86. return null;
  87. }
  88. public function __construct( $blog_id, $preview ) {
  89. $this->_blog_id = $blog_id;
  90. $this->_preview = $preview;
  91. include W3TC_DIR . '/ConfigKeys.php';
  92. $this->_keys = $keys;
  93. // move _date to initial state
  94. foreach ( $this->_keys as $key => $value )
  95. $this->_data[$key] = $value['default'];
  96. $this->_data['version'] = W3TC_VERSION;
  97. }
  98. public function load() {
  99. // apply data from master config
  100. $master_filename = Config::util_config_filename( 0, $this->_preview );
  101. $data = Config::util_array_from_file( $master_filename );
  102. if ( is_null( $data ) && $this->_preview ) {
  103. // try to read production data when preview not available
  104. $master_filename = Config::util_config_filename( 0, false );
  105. $data = Config::util_array_from_file( $master_filename );
  106. }
  107. // try to get legacy v2 data
  108. if ( is_null( $data ) ) {
  109. $master_filename = Config::util_config_filename_legacy_v2( 0,
  110. $this->_preview );
  111. $data = self::util_array_from_file_legacy_v2( $master_filename );
  112. }
  113. // try to get legacy v1 data
  114. if ( is_null( $data ) ) {
  115. $master_filename = Config::util_config_filename_legacy_v1( 0,
  116. $this->_preview );
  117. $data = self::util_array_from_file_legacy_v1( $master_filename );
  118. }
  119. if ( is_array( $data ) ) {
  120. $data = $this->upgrade( $data );
  121. foreach ( $data as $key => $value )
  122. $this->_data[$key] = $value;
  123. }
  124. if ( $this->is_master() )
  125. return;
  126. // apply child config
  127. $child_filename = Config::util_config_filename( $this->_blog_id,
  128. $this->_preview );
  129. $data = Config::util_array_from_file( $child_filename );
  130. if ( is_null( $data ) && $this->_preview ) {
  131. // try to read production data when preview not available
  132. $child_filename = Config::util_config_filename( $this->_blog_id,
  133. false );
  134. $data = Config::util_array_from_file( $child_filename );
  135. }
  136. // try to get legacy v2 data
  137. if ( is_null( $data ) ) {
  138. $child_filename = Config::util_config_filename_legacy_v2(
  139. $this->_blog_id, $this->_preview );
  140. $data = self::util_array_from_file_legacy_v2( $child_filename );
  141. }
  142. // try to get legacy v1 data
  143. if ( is_null( $data ) ) {
  144. $child_filename = Config::util_config_filename_legacy_v1(
  145. $this->_blog_id, $this->_preview );
  146. $data = self::util_array_from_file_legacy_v1( $child_filename );
  147. }
  148. if ( is_array( $data ) ) {
  149. $data = $this->upgrade( $data );
  150. foreach ( $data as $key => $value ) {
  151. if ( !ConfigCompiler::child_key_sealed( $key, $this->_data, $data ) )
  152. $this->_data[$key] = $value;
  153. }
  154. }
  155. }
  156. public function apply_data( $data ) {
  157. foreach ( $data as $key => $value )
  158. $this->_data[$key] = $value;
  159. }
  160. public function get_data() {
  161. return $this->_data;
  162. }
  163. public function save() {
  164. $data = array(
  165. 'version' => $this->_data['version']
  166. );
  167. if ( $this->is_master() ) {
  168. foreach ( $this->_data as $key => $value )
  169. $data[$key] = $this->_data[$key];
  170. } else {
  171. // write only overwrited keys
  172. $master = new ConfigCompiler( 0, $this->_preview );
  173. $master->load();
  174. foreach ( $this->_data as $key => $value ) {
  175. if ( !ConfigCompiler::child_key_sealed( $key, $master->_data,
  176. $this->_data ) )
  177. $data[$key] = $this->_data[$key];
  178. }
  179. }
  180. $filename = Config::util_config_filename( $this->_blog_id,
  181. $this->_preview );
  182. if ( defined( 'JSON_PRETTY_PRINT' ) )
  183. $config = json_encode( $data, JSON_PRETTY_PRINT );
  184. else // for older php versions
  185. $config = json_encode( $data );
  186. Util_File::file_put_contents_atomic( $filename, '<?php exit; ?>' . $config );
  187. }
  188. /**
  189. * Returns true if we edit master config
  190. *
  191. * @return boolean
  192. */
  193. private function is_master() {
  194. return $this->_blog_id <= 0;
  195. }
  196. /**
  197. * Apply new default values when version changes
  198. */
  199. private function upgrade( $file_data ) {
  200. if ( !isset( $file_data['version'] ) )
  201. $file_data['version'] = '0.0.0';
  202. if ( !function_exists( 'bb2_start' ) ) {
  203. $file_data['pgcache.bad_behavior_path'] = '';
  204. } else {
  205. if ( file_exists( WP_PLUGIN_DIR . '/bad-behavior/bad-behavior-generic.php' ) ) {
  206. $bb_file = WP_PLUGIN_DIR . '/bad-behavior/bad-behavior-generic.php';
  207. } elseif ( file_exists( WP_PLUGIN_DIR . '/Bad-Behavior/bad-behavior-generic.php' ) ) {
  208. $bb_file = WP_PLUGIN_DIR . '/Bad-Behavior/bad-behavior-generic.php';
  209. } else {
  210. $bb_file = false;
  211. }
  212. if ( $bb_file ) {
  213. $file_data['pgcache.bad_behavior_path'] = $bb_file;
  214. }
  215. }
  216. //
  217. // changes in 0.9.5
  218. //
  219. if ( !isset( $file_data['extensions.active_frontend'] ) ||
  220. !is_array( $file_data['extensions.active_frontend'] ) )
  221. $file_data['extensions.active_frontend'] = array();
  222. if ( version_compare( $file_data['version'], '0.9.5', '<' ) ) {
  223. // dont show minify tips if already enabled
  224. if ( isset( $file_data['minify.enabled'] ) &&
  225. $file_data['minify.enabled'] == 'true' &&
  226. function_exists( 'get_option' ) ) {
  227. $cs = Dispatcher::config_state();
  228. $cs->set( 'minify.hide_minify_help', true );
  229. $cs->save();
  230. }
  231. $file_data['pgcache.mirrors.enabled'] = true;
  232. // map regions in rackspace
  233. if ( isset( $file_data['cdn.rscf.location'] ) ) {
  234. if ( $file_data['cdn.rscf.location'] == 'uk' )
  235. $file_data['cdn.rscf.location'] = 'LON';
  236. if ( $file_data['cdn.rscf.location'] == 'us' )
  237. $file_data['cdn.rscf.location'] = 'ORD';
  238. }
  239. // change filenames
  240. $active = array();
  241. if ( isset( $file_data['extensions.active'] ) &&
  242. is_array( $file_data['extensions.active'] ) ) {
  243. if ( isset( $file_data['extensions.active']['cloudflare'] ) )
  244. $active['cloudflare'] = 'w3-total-cache/Extension_CloudFlare_Plugin.php';
  245. if ( isset( $file_data['extensions.active']['feedburner'] ) )
  246. $active['feedburner'] = 'w3-total-cache/Extension_FeedBurner_Plugin.php';
  247. if ( isset( $file_data['extensions.active']['genesis.theme'] ) )
  248. $active['genesis.theme'] = 'w3-total-cache/Extension_Genesis_Plugin.php';
  249. if ( isset( $file_data['extensions.active']['wordpress-seo'] ) )
  250. $active['wordpress-seo'] = 'w3-total-cache/Extension_WordPressSeo_Plugin.php';
  251. }
  252. $file_data['extensions.active'] = $active;
  253. $active_frontend = array();
  254. foreach ( $active as $key => $value )
  255. $active_frontend[$key] = '*';
  256. $file_data['extensions.active_frontend'] = $active_frontend;
  257. // keep those active by default
  258. $file_data['extensions.active']['newrelic'] =
  259. 'w3-total-cache/Extension_NewRelic_Plugin.php';
  260. $file_data['extensions.active']['fragmentcache'] =
  261. 'w3-total-cache/Extension_FragmentCache_Plugin.php';
  262. }
  263. // newrelic settings - migrate to extension
  264. if ( isset( $file_data['newrelic.enabled'] ) &&
  265. $file_data['newrelic.enabled'] ) {
  266. // make new relic extension enabled
  267. if ( !isset( $file_data['extensions.active_frontend']['newrelic'] ) )
  268. $file_data['extensions.active_frontend']['newrelic'] ='*';
  269. }
  270. if ( !isset( $file_data['newrelic'] ) ||
  271. !is_array( $file_data['newrelic'] ) )
  272. $file_data['newrelic'] = array(
  273. 'monitoring_type' => 'apm'
  274. );
  275. $this->_set_if_exists( $file_data, 'newrelic.api_key',
  276. 'newrelic', 'api_key' );
  277. $this->_set_if_exists( $file_data, 'newrelic.appname',
  278. 'newrelic', 'apm.application_name' );
  279. $this->_set_if_exists( $file_data, 'newrelic.accept.logged_roles',
  280. 'newrelic', 'accept.logged_roles' );
  281. $this->_set_if_exists( $file_data, 'newrelic.accept.roles',
  282. 'newrelic', 'accept.roles' );
  283. $this->_set_if_exists( $file_data, 'newrelic.use_php_function',
  284. 'newrelic', 'use_php_function' );
  285. $this->_set_if_exists( $file_data, 'newrelic.cache_time',
  286. 'newrelic', 'cache_time' );
  287. $this->_set_if_exists( $file_data, 'newrelic.enable_xmit',
  288. 'newrelic', 'enable_xmit' );
  289. $this->_set_if_exists( $file_data, 'newrelic.include_rum',
  290. 'newrelic', 'include_rum' );
  291. // extensions - kept in separate key now
  292. $this->_set_if_exists_extension( $file_data, 'cloudflare' );
  293. $this->_set_if_exists_extension( $file_data, 'genesis.theme' );
  294. $this->_set_if_exists_extension( $file_data, 'feedburner' );
  295. // fragmentcache to extension
  296. if ( isset( $file_data['fragmentcache.enabled'] ) &&
  297. $file_data['fragmentcache.enabled'] ) {
  298. // make new relic extension enabled
  299. if ( !isset( $file_data['extensions.active_frontend']['fragmentcache'] ) )
  300. $file_data['extensions.active_frontend']['fragmentcache'] = '*';
  301. }
  302. $this->_set_if_exists( $file_data, 'fragmentcache.debug',
  303. 'fragmentcache', 'debug' );
  304. $this->_set_if_exists( $file_data, 'fragmentcache.engine',
  305. 'fragmentcache', 'engine' );
  306. $this->_set_if_exists( $file_data, 'fragmentcache.file.gc',
  307. 'fragmentcache', 'file.gc' );
  308. $this->_set_if_exists( $file_data, 'fragmentcache.file.locking',
  309. 'fragmentcache', 'file.locking' );
  310. $this->_set_if_exists( $file_data, 'fragmentcache.memcached.servers',
  311. 'fragmentcache', 'memcached.servers' );
  312. $this->_set_if_exists( $file_data, 'fragmentcache.memcached.persistent',
  313. 'fragmentcache', 'memcached.persistent' );
  314. $this->_set_if_exists( $file_data, 'fragmentcache.memcached.aws_autodiscovery',
  315. 'fragmentcache', 'memcached.aws_autodiscovery' );
  316. $this->_set_if_exists( $file_data, 'fragmentcache.memcached.username',
  317. 'fragmentcache', 'memcached.username' );
  318. $this->_set_if_exists( $file_data, 'fragmentcache.memcached.password',
  319. 'fragmentcache', 'memcached.password' );
  320. $this->_set_if_exists( $file_data, 'fragmentcache.redis.persistent',
  321. 'fragmentcache', 'redis.persistent' );
  322. $this->_set_if_exists( $file_data, 'fragmentcache.redis.servers',
  323. 'fragmentcache', 'redis.servers' );
  324. $this->_set_if_exists( $file_data, 'fragmentcache.redis.password',
  325. 'fragmentcache', 'redis.password' );
  326. $this->_set_if_exists( $file_data, 'fragmentcache.redis.dbid',
  327. 'fragmentcache', 'redis.dbid' );
  328. $this->_set_if_exists( $file_data, 'fragmentcache.lifetime',
  329. 'fragmentcache', 'lifetime' );
  330. // new options, separated old one. implemented in 0.9.5.3
  331. if ( isset( $file_data['browsercache.cssjs.replace'] ) &&
  332. !isset( $file_data['browsercache.cssjs.querystring'] ) ) {
  333. $file_data['browsercache.cssjs.querystring'] = $file_data['browsercache.cssjs.replace'];
  334. }
  335. if ( isset( $file_data['browsercache.other.replace'] ) &&
  336. !isset( $file_data['browsercache.other.querystring'] ) ) {
  337. $file_data['browsercache.other.querystring'] = $file_data['browsercache.other.replace'];
  338. }
  339. //
  340. // changes in 0.9.5.3
  341. //
  342. if ( version_compare( $file_data['version'], '0.9.5.3', '<' ) ) {
  343. if ( !isset( $file_data['extensions.active']['swarmify'] ) ) {
  344. $file_data['extensions.active']['swarmify'] = 'w3-total-cache/Extension_Swarmify_Plugin.php';
  345. }
  346. }
  347. $file_data['version'] = W3TC_VERSION;
  348. return $file_data;
  349. }
  350. private function _set_if_exists_extension( &$a, $extension ) {
  351. if ( isset( $a['extensions.settings'] ) &&
  352. isset( $a['extensions.settings'][$extension] ) ) {
  353. $a[$extension] = $a['extensions.settings'][$extension];
  354. unset( $a['extensions.settings'][$extension] );
  355. }
  356. }
  357. private function _set_if_exists( &$a, $old_key, $new_key0, $new_key1 ) {
  358. if ( isset( $a[$old_key] ) ) {
  359. $a[$new_key0][$new_key1] = $a[$old_key];
  360. unset( $a[$old_key] );
  361. }
  362. }
  363. }