|
| 1 | +<?php |
| 2 | + |
| 3 | +/** |
| 4 | + * Class CRM_Core_Smarty_plugins_UrlTest |
| 5 | + * @group headless |
| 6 | + */ |
| 7 | +class CRM_Core_Smarty_plugins_UrlTest extends CiviUnitTestCase { |
| 8 | + |
| 9 | + public function setUp(): void { |
| 10 | + parent::setUp(); |
| 11 | + require_once 'CRM/Core/Smarty.php'; |
| 12 | + |
| 13 | + // Templates should normally be file names, but for unit-testing it's handy to use "string:" notation |
| 14 | + require_once 'CRM/Core/Smarty/resources/String.php'; |
| 15 | + civicrm_smarty_register_string_resource(); |
| 16 | + |
| 17 | + $this->useTransaction(); |
| 18 | + } |
| 19 | + |
| 20 | + /** |
| 21 | + * @return array |
| 22 | + */ |
| 23 | + public function urlCases() { |
| 24 | + $literal = function(string $s) { |
| 25 | + return '!' . preg_quote($s, '!') . '!'; |
| 26 | + }; |
| 27 | + |
| 28 | + $cases = []; |
| 29 | + $cases[] = [ |
| 30 | + // Generate an ordinary, HTML-style URL. |
| 31 | + $literal('q=civicrm/profile/view&id=123&gid=456'), |
| 32 | + '{url}//civicrm/profile/view?id=123&gid=456{/url}', |
| 33 | + ]; |
| 34 | + $cases[] = [ |
| 35 | + // Here, we assign the plain-text variable and then use it for JS expression |
| 36 | + '!window.location = ".*q=civicrm/profile/view&id=123&gid=456"!', |
| 37 | + '{url assign=myUrl flags=t}//civicrm/profile/view?id=123&gid=456{/url}' . |
| 38 | + 'window.location = "{$myUrl}";', |
| 39 | + ]; |
| 40 | + $cases[] = [ |
| 41 | + $literal('q=civicrm/profile/view&id=999&message=hello+world'), |
| 42 | + '{url 1="999" 2="hello world"}//civicrm/profile/view?id=[1]&message=[2]{/url}', |
| 43 | + ]; |
| 44 | + $cases[] = [ |
| 45 | + $literal('q=civicrm/profile/view&id=123&message=hello+world'), |
| 46 | + '{url msg="hello world"}//civicrm/profile/view?id=123&message=[msg]{/url}', |
| 47 | + ]; |
| 48 | + $cases[] = [ |
| 49 | + // Define a temporary variable for use in the URL. |
| 50 | + $literal('q=civicrm/profile/view&id=123&message=this+%26+that'), |
| 51 | + '{url msg="this & that"}//civicrm/profile/view?id=123&message=[msg]{/url}', |
| 52 | + ]; |
| 53 | + $cases[] = [ |
| 54 | + // We have a Smarty variable which already included escaped data. Smarty should do substitution. |
| 55 | + $literal('q=civicrm/profile/view&id=123&message=this+%2B+that'), |
| 56 | + '{assign var=msg value="this+%2B+that"}' . |
| 57 | + '{url flags=%}//civicrm/profile/view?id=123&message={$msg}{/url}', |
| 58 | + ]; |
| 59 | + $cases[] = [ |
| 60 | + // Generate client-side route (with Angular path and params) |
| 61 | + $literal('q=civicrm/a/#/mailing/100?angularDebug=1'), |
| 62 | + '{url id=100}backend://civicrm/a/#/mailing/[id]?angularDebug=1{/url}', |
| 63 | + ]; |
| 64 | + |
| 65 | + // This example is neat - you just replace `{$msg}` with `[msg]`, and then you get encoded URL data. |
| 66 | + // But... it's pretty shallow. You can't use Smarty expressions or modifiers. Additionally, |
| 67 | + // enabling this mode increases the risk of accidental collisions between Smarty variables |
| 68 | + // and deep-form-params. So I've left it disabled for now. |
| 69 | + // |
| 70 | + // $cases[] = [ |
| 71 | + // // We have a Smarty variable with canonical (unescaped) data. Use it as URL variable. |
| 72 | + // $literal('q=civicrm/profile/view&id=123&message=this+%2B+that'), |
| 73 | + // '{assign var=msg value="this + that"}' . |
| 74 | + // '{url}//civicrm/profile/view?id=123&message=[msg]{/url}', |
| 75 | + // ]; |
| 76 | + |
| 77 | + // return CRM_Utils_Array::subset($cases, [2]); |
| 78 | + return $cases; |
| 79 | + } |
| 80 | + |
| 81 | + /** |
| 82 | + * @dataProvider urlCases |
| 83 | + * @param string $expected |
| 84 | + * @param string $input |
| 85 | + */ |
| 86 | + public function testUrl($expected, $input) { |
| 87 | + $smarty = CRM_Core_Smarty::singleton(); |
| 88 | + $actual = $smarty->fetch('string:' . $input); |
| 89 | + $this->assertRegExp($expected, $actual, "Process input=[$input]"); |
| 90 | + } |
| 91 | + |
| 92 | +} |
0 commit comments