Skip to content

Commit c9f920f

Browse files
authored
Merge pull request #94 from iMattPro/sanitization
Try again at shortname cleanup
2 parents 89eda11 + dd44ba1 commit c9f920f

File tree

6 files changed

+87
-20
lines changed

6 files changed

+87
-20
lines changed

controller/manifest.php

+7-2
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
use phpbb\language\language;
1616
use phpbb\path_helper;
1717
use phpbb\user;
18+
use phpbb\webpushnotifications\ext;
1819
use Symfony\Component\HttpFoundation\JsonResponse;
1920
use Symfony\Component\HttpFoundation\Response;
2021

@@ -63,9 +64,13 @@ public function handle(): JsonResponse
6364
$board_path = $this->config['force_server_vars'] ? $this->config['script_path'] : $this->path_helper->get_web_root_path();
6465
$board_url = generate_board_url();
6566

67+
// Emoji fixer-uppers
68+
$sitename = ext::decode_entities($this->config['sitename'], ENT_QUOTES);
69+
$pwa_short_name = ext::decode_entities($this->config['pwa_short_name'], ENT_QUOTES);
70+
6671
$manifest = [
67-
'name' => $this->config['sitename'],
68-
'short_name' => $this->config['pwa_short_name'] ?: utf8_substr(preg_replace('/[^\x21-\x7E]/', '', html_entity_decode($this->config['sitename'], ENT_QUOTES, 'UTF-8')), 0, 12),
72+
'name' => $sitename,
73+
'short_name' => $pwa_short_name ?: utf8_substr($sitename, 0, 12),
6974
'display' => 'standalone',
7075
'orientation' => 'portrait',
7176
'dir' => $this->language->lang('DIRECTION'),

event/listener.php

+29-14
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
use phpbb\notification\manager;
1818
use phpbb\template\template;
1919
use phpbb\user;
20+
use phpbb\webpushnotifications\ext;
2021
use phpbb\webpushnotifications\form\form_helper;
2122
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
2223

@@ -85,6 +86,7 @@ public static function getSubscribedEvents()
8586
'core.ucp_display_module_before' => 'load_language',
8687
'core.acp_main_notice' => 'compatibility_notice',
8788
'core.acp_board_config_edit_add' => 'acp_pwa_options',
89+
'core.acp_board_config_emoji_enabled'=> 'acp_pwa_allow_emoji',
8890
'core.validate_config_variable' => 'validate_pwa_options',
8991
'core.help_manager_add_block_after' => 'wpn_faq',
9092
];
@@ -143,7 +145,7 @@ public function pwa_manifest()
143145
$this->template->assign_vars([
144146
'U_MANIFEST_URL' => $this->controller_helper->route('phpbb_webpushnotifications_manifest_controller'),
145147
'U_TOUCH_ICON' => $this->config['pwa_icon_small'],
146-
'SHORT_SITE_NAME' => $this->config['pwa_short_name'] ?: $this->get_shortname($this->config['sitename']),
148+
'SHORT_SITE_NAME' => $this->config['pwa_short_name'] ?: $this->trim_shortname($this->config['sitename']),
147149
]);
148150
}
149151

@@ -170,6 +172,24 @@ public function acp_pwa_options($event)
170172
}
171173
}
172174

175+
/**
176+
* Allow PWA short name ACP field to accept emoji characters
177+
*
178+
* @param \phpbb\event\data $event
179+
* @return void
180+
*/
181+
public function acp_pwa_allow_emoji($event)
182+
{
183+
if (in_array('pwa_short_name', $event['config_name_ary'], true))
184+
{
185+
return;
186+
}
187+
188+
$config_name_ary = $event['config_name_ary'];
189+
$config_name_ary[] = 'pwa_short_name';
190+
$event['config_name_ary'] = $config_name_ary;
191+
}
192+
173193
/**
174194
* Return HTML for PWA icon name settings
175195
*
@@ -191,7 +211,7 @@ public function pwa_icon_name($value, $key)
191211
*/
192212
public function pwa_short_sitename($value, $key)
193213
{
194-
$placeholder = $this->get_shortname($this->config['sitename']);
214+
$placeholder = $this->trim_shortname($this->config['sitename']);
195215

196216
return '<input id="' . $key . '" type="text" size="40" maxlength="12" name="config[' . $key . ']" value="' . $value . '" placeholder="' . $placeholder . '">';
197217
}
@@ -223,17 +243,10 @@ public function validate_pwa_options($event)
223243
return;
224244
}
225245

226-
$short_name = $event['cfg_array']['pwa_short_name'];
227-
228-
// Do not allow multibyte characters or emoji
229-
if (strlen($short_name) !== mb_strlen($short_name, 'UTF-8'))
230-
{
231-
$this->add_error($event, 'PWA_SHORT_NAME_INVALID');
232-
return;
233-
}
246+
$short_name = ext::decode_entities($event['cfg_array']['pwa_short_name'], ENT_QUOTES);
234247

235248
// Do not allow strings longer than 12 characters
236-
if (strlen($short_name) > 12)
249+
if (utf8_strlen($short_name) > 12)
237250
{
238251
$this->add_error($event, 'PWA_SHORT_NAME_INVALID');
239252
return;
@@ -346,13 +359,15 @@ protected function can_use_notifications()
346359
}
347360

348361
/**
349-
* Get short name from a string (strip out multibyte characters and trim to 12 characters)
362+
* Trim short name from a string to 12 characters
350363
*
351364
* @param string $name
352365
* @return string 12 max characters string
353366
*/
354-
protected function get_shortname($name)
367+
protected function trim_shortname($name)
355368
{
356-
return utf8_substr(preg_replace('/[^\x20-\x7E]/', '', $name), 0, 12);
369+
$decoded = ext::decode_entities($name, ENT_QUOTES);
370+
$trimmed = utf8_substr($decoded, 0, 12);
371+
return htmlspecialchars($trimmed, ENT_QUOTES, 'UTF-8');
357372
}
358373
}

ext.php

+13
Original file line numberDiff line numberDiff line change
@@ -124,4 +124,17 @@ protected function result()
124124

125125
return false;
126126
}
127+
128+
/**
129+
* Decode entities, used primarily to fix emoji for display
130+
*
131+
* @param string $text
132+
* @param int $flags Uses ENT_NOQUOTES to leave single and double quotes encoded by default
133+
* @param string $encoding
134+
* @return string Decoded string
135+
*/
136+
public static function decode_entities($text, $flags = ENT_NOQUOTES, $encoding = 'UTF-8')
137+
{
138+
return html_entity_decode($text, $flags, $encoding);
139+
}
127140
}

language/en/webpushnotifications_common_acp.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,7 @@
4141
'PWA_SETTINGS' => 'Progressive web application options',
4242
'PWA_SHORT_NAME' => 'Short site name',
4343
'PWA_SHORT_NAME_EXPLAIN' => 'Your site name in 12 characters or fewer, which may be used as a label for an icon on a mobile device’s home screen. (If this field is left empty, the first 12 characters of the <samp>Site name</samp> will be used.)',
44-
'PWA_SHORT_NAME_INVALID' => '“Short site name” contains illegal characters or exceeds the 12 character limit.',
44+
'PWA_SHORT_NAME_INVALID' => '“Short site name” exceeds the 12 character limit.',
4545
'PWA_ICON_SMALL' => 'Small mobile device icon',
4646
'PWA_ICON_SMALL_EXPLAIN' => 'File name of a 192px x 192px PNG image. This file must be uploaded to your board’s <samp>icons</samp> directory.',
4747
'PWA_ICON_LARGE' => 'Large mobile device icon',

tests/event/listener_test.php

+34-1
Original file line numberDiff line numberDiff line change
@@ -149,6 +149,7 @@ public function test_getSubscribedEvents()
149149
'core.ucp_display_module_before',
150150
'core.acp_main_notice',
151151
'core.acp_board_config_edit_add',
152+
'core.acp_board_config_emoji_enabled',
152153
'core.validate_config_variable',
153154
'core.help_manager_add_block_after',
154155
], array_keys(\phpbb\webpushnotifications\event\listener::getSubscribedEvents()));
@@ -376,7 +377,6 @@ public function test_acp_pwa_options($mode, $display_vars, $expected_keys)
376377
$keys = array_keys($display_vars['vars']);
377378

378379
self::assertEquals($expected_keys, $keys);
379-
380380
}
381381

382382
public function validate_pwa_options_data()
@@ -420,6 +420,21 @@ public function validate_pwa_options_data()
420420
[
421421
'pwa_options:string',
422422
['pwa_short_name' => 'foo❤️'],
423+
[],
424+
],
425+
[
426+
'pwa_options:string',
427+
['pwa_short_name' => 'Фаны phpBB'],
428+
[],
429+
],
430+
[
431+
'pwa_options:string',
432+
['pwa_short_name' => 'Фаны phpBB Board'],
433+
['PWA_SHORT_NAME_INVALID'],
434+
],
435+
[
436+
'pwa_options:string',
437+
['pwa_short_name' => 'foo❤️bar foo bar'],
423438
['PWA_SHORT_NAME_INVALID'],
424439
],
425440
[
@@ -472,6 +487,24 @@ public function test_validate_pwa_options($validate, $cfg_array, $expected_error
472487
self::assertEquals($expected_error, $error);
473488
}
474489

490+
public function test_acp_pwa_allow_emoji()
491+
{
492+
$config_name_ary = ['foo'];
493+
$expected = ['foo', 'pwa_short_name'];
494+
495+
$this->set_listener();
496+
497+
$dispatcher = new \phpbb\event\dispatcher();
498+
$dispatcher->addListener('core.acp_board_config_emoji_enabled', [$this->listener, 'acp_pwa_allow_emoji']);
499+
500+
$event_data = ['config_name_ary'];
501+
$event_data_after = $dispatcher->trigger_event('core.acp_board_config_emoji_enabled', compact($event_data));
502+
503+
extract($event_data_after, EXTR_OVERWRITE);
504+
505+
self::assertEquals($expected, $config_name_ary);
506+
}
507+
475508
public function test_wpn_faq()
476509
{
477510
$this->language->add_lang('webpushnotifications_faq', 'phpbb/webpushnotifications');

ucp/controller/webpush.php

+3-2
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
use phpbb\exception\http_exception;
1717
use phpbb\language\language;
1818
use phpbb\notification\manager;
19+
use phpbb\webpushnotifications\ext;
1920
use phpbb\webpushnotifications\form\form_helper;
2021
use phpbb\webpushnotifications\json\sanitizer as json_sanitizer;
2122
use phpbb\path_helper;
@@ -240,8 +241,8 @@ private function get_notification_data(string $notification_data): string
240241

241242
return json_encode([
242243
'heading' => $this->config['sitename'],
243-
'title' => strip_tags(html_entity_decode($notification->get_title(), ENT_NOQUOTES, 'UTF-8')),
244-
'text' => strip_tags(html_entity_decode($notification->get_reference(), ENT_NOQUOTES, 'UTF-8')),
244+
'title' => strip_tags(ext::decode_entities($notification->get_title())),
245+
'text' => strip_tags(ext::decode_entities($notification->get_reference())),
245246
'url' => htmlspecialchars_decode($notification->get_url()),
246247
'avatar' => $this->prepare_avatar($notification->get_avatar()),
247248
]);

0 commit comments

Comments
 (0)