From 9996d56dbebec6183c85856c0351b0d4cbb1bf8a Mon Sep 17 00:00:00 2001 From: Tomas Dittmann Date: Mon, 23 Oct 2023 16:38:44 +0200 Subject: [PATCH 1/7] use html dataset find checkboxes and their offsets more reliably add test markdown add comments for future expansion nummerize checkboxes from top to bottom - the markdown integrated way was dependend on nesting --- Assets/js/checkbox.js | 4 +- Controller/CheckboxController.php | 118 +++++++++++------- Helper/CoreMarkdown.php | 37 +++++- .../parsedown-checkbox/ParsedownCheckbox.php | 4 +- 4 files changed, 108 insertions(+), 55 deletions(-) diff --git a/Assets/js/checkbox.js b/Assets/js/checkbox.js index 589830a..9d615de 100644 --- a/Assets/js/checkbox.js +++ b/Assets/js/checkbox.js @@ -23,10 +23,10 @@ KB.on('dom.ready', function () { else { link = '?controller=CheckboxController&action=toggle&plugin=MarkdownPlus'; } - + KB.http.postJson(link, { 'task_id': task_id, - 'number': e.target.getAttribute('number') + 'number': e.target.dataset.number }); } } diff --git a/Controller/CheckboxController.php b/Controller/CheckboxController.php index 1c53e05..c50b5b5 100644 --- a/Controller/CheckboxController.php +++ b/Controller/CheckboxController.php @@ -7,7 +7,29 @@ class CheckboxController extends BaseController { - private const regexCheckbox = '/\[[x, ]\]/m'; + /* + # must match + + [x] 2 + * [ ] 3 + > - [ ] 1 + + [x] 2 + * [ ] 3 + > > * * [ ] 3 + > * > * * [ ] 3 + [ ] test 8 + + # must not match: + + - [x]1 + + [ ] + test [x] + test [x] + [] test 3 + [a] test + [x](www.google.de) test4 + */ + + private const regexCheckbox = '/^([+,\-,*,>, ] )*(\[[x, ]\] )/m'; private function findCheckBox($text, $number, &$offset) { @@ -17,7 +39,9 @@ private function findCheckBox($text, $number, &$offset) if ($offset >= $number) { return array( 'success' => true, - 'offset' => $matches[0][$count - $offset + $number - 1][1] + 1 + 'offset' => end($matches) // the actual box [ ]/[x] + [$count - $offset + $number - 1] // the requested checkbox + [1] + 1 // the offset to the checkbox content ); } return array('success' => false); @@ -35,69 +59,71 @@ private function togglechar(&$string, $offset) public function toggle() { $values = $this->request->getJson(); - - $foundCheckboxes = 0; - $number = intval($values['number']); $taskId = $values['task_id']; if (isset($taskId)) { $task = $this->taskFinderModel->getById($taskId); - $text = $task['description']; - $result = $this->findCheckBox($text, $number, $foundCheckboxes); - if ($result['success']) { - $this->togglechar($text, $result['offset']); - $task['description'] = $text; - $this->taskModificationModel->update($task); - return; - } + if ($task) { + $foundCheckboxes = 0; + $number = intval($values['number']); - if (isset($this->container["definitionOfDoneModel"])) { - foreach ($this->definitionOfDoneModel->getAll($taskId) as $subtask) { - $dod = $this->definitionOfDoneModel->getById($subtask['id']); + $text = $task['description']; + $result = $this->findCheckBox($text, $number, $foundCheckboxes); + if ($result['success']) { + $this->togglechar($text, $result['offset']); + $task['description'] = $text; + $this->taskModificationModel->update($task); + return; + } - $result = $this->findCheckBox($dod['title'], $number, $foundCheckboxes); - if ($result['success']) { - $this->togglechar($dod['title'], $result['offset']); - $this->definitionOfDoneModel->save($dod); - return; + if (isset($this->container["definitionOfDoneModel"])) { + foreach ($this->definitionOfDoneModel->getAll($taskId) as $subtask) { + $dod = $this->definitionOfDoneModel->getById($subtask['id']); + + $result = $this->findCheckBox($dod['title'], $number, $foundCheckboxes); + if ($result['success']) { + $this->togglechar($dod['title'], $result['offset']); + $this->definitionOfDoneModel->save($dod); + return; + } + + $result = $this->findCheckBox($dod['text'], $number, $foundCheckboxes); + if ($result['success']) { + $this->togglechar($dod['text'], $result['offset']); + $this->definitionOfDoneModel->save($dod); + return; + } } + } - $result = $this->findCheckBox($dod['text'], $number, $foundCheckboxes); - if ($result['success']) { - $this->togglechar($dod['text'], $result['offset']); - $this->definitionOfDoneModel->save($dod); - return; + if (isset($this->container["subtaskResultModel"])) { + foreach ($this->subtaskModel->getAll($taskId) as $subtask) { + $text = $this->subtaskResultModel->getById($subtask['id']); + $result = $this->findCheckBox($text, $number, $foundCheckboxes); + if ($result['success']) { + $this->togglechar($text, $result['offset']); + $this->subtaskResultModel->Save($subtask['id'], $text); + return; + } } } - } - if (isset($this->container["subtaskResultModel"])) { - foreach ($this->subtaskModel->getAll($taskId) as $subtask) { - $text = $this->subtaskResultModel->getById($subtask['id']); + $commentSortingDirection = $this->userMetadataCacheDecorator->get(UserMetadataModel::KEY_COMMENT_SORTING_DIRECTION, 'ASC'); + + foreach ($this->commentModel->getAll($taskId, $commentSortingDirection) as $comment) { + $text = $comment['comment']; + $result = $this->findCheckBox($text, $number, $foundCheckboxes); + if ($result['success']) { $this->togglechar($text, $result['offset']); - $this->subtaskResultModel->Save($subtask['id'], $text); + $comment['comment'] = $text; + $this->commentModel->update($comment); return; } } } - - $commentSortingDirection = $this->userMetadataCacheDecorator->get(UserMetadataModel::KEY_COMMENT_SORTING_DIRECTION, 'ASC'); - - foreach ($this->commentModel->getAll($taskId, $commentSortingDirection) as $comment) { - $text = $comment['comment']; - - $result = $this->findCheckBox($text, $number, $foundCheckboxes); - - if ($result['success']) { - $this->togglechar($text, $result['offset']); - $comment['comment'] = $text; - $this->commentModel->update($comment); - return; - } - } } } } diff --git a/Helper/CoreMarkdown.php b/Helper/CoreMarkdown.php index 87878dc..9fbf9d5 100755 --- a/Helper/CoreMarkdown.php +++ b/Helper/CoreMarkdown.php @@ -63,7 +63,7 @@ protected function inlineTaskLink(array $Excerpt) if (preg_match('!#(\d+)!i', $Excerpt['text'], $matches)) { $link = $this->buildTaskLink($matches[1]); - if (! empty($link)) { + if (!empty($link)) { return array( 'extent' => strlen($matches[0]), 'element' => array( @@ -89,11 +89,11 @@ protected function inlineTaskLink(array $Excerpt) */ protected function inlineUserLink(array $Excerpt) { - if (! $this->isPublicLink && preg_match('/^@([^\s,!:?]+)/', $Excerpt['text'], $matches)) { + if (!$this->isPublicLink && preg_match('/^@([^\s,!:?]+)/', $Excerpt['text'], $matches)) { $username = rtrim($matches[1], '.'); $user = $this->container['userCacheDecorator']->getByUsername($username); - if (! empty($user)) { + if (!empty($user)) { $url = $this->container['helper']->url->to('UserViewController', 'profile', array('user_id' => $user['id'])); $name = $user['name'] ?: $user['username']; @@ -128,7 +128,7 @@ private function buildTaskLink($task_id) if ($this->isPublicLink) { $token = $this->container['memoryCache']->proxy($this->container['taskFinderModel'], 'getProjectToken', $task_id); - if (! empty($token)) { + if (!empty($token)) { return $this->container['helper']->url->to( 'TaskViewController', 'readonly', @@ -165,4 +165,31 @@ protected function inlineLink($Excerpt) } return $Inline; } -} \ No newline at end of file + + function text($text) + { + $markup = parent::text($text); + + $this->nummerizeCheckboxes($markup); + + return $markup; + } + + private function nummerizeCheckboxes(&$markup) + { + $counter = new Counter; + $count = 0; + + $markup = preg_replace_callback("/activecheckbox\"/m", $counter->count(...), $markup, -1, $count, PREG_OFFSET_CAPTURE); + } +} + +class Counter +{ + private $count = 0; + + public function count($matches) { + $this->count++; + return $matches[0][0] . " data-number=". $this->count . " "; + } +}; \ No newline at end of file diff --git a/vendor/leblanc-simon/parsedown-checkbox/ParsedownCheckbox.php b/vendor/leblanc-simon/parsedown-checkbox/ParsedownCheckbox.php index f8781d8..6d8e548 100644 --- a/vendor/leblanc-simon/parsedown-checkbox/ParsedownCheckbox.php +++ b/vendor/leblanc-simon/parsedown-checkbox/ParsedownCheckbox.php @@ -74,7 +74,7 @@ protected function checkboxUnchecked($text) $text = self::escape($text); } - return ' ' . $this->format($text); + return ' ' . $this->format($text); } protected function checkboxChecked($text) @@ -83,7 +83,7 @@ protected function checkboxChecked($text) $text = self::escape($text); } - return ' ' . $this->format($text); + return ' ' . $this->format($text); } /** From 6b4401ee8ef79ac3296a683c14ea41d2b26fed32 Mon Sep 17 00:00:00 2001 From: Tomas Dittmann Date: Tue, 24 Oct 2023 12:06:46 +0200 Subject: [PATCH 2/7] use php 7 compatible syntax --- Helper/CoreMarkdown.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Helper/CoreMarkdown.php b/Helper/CoreMarkdown.php index 9fbf9d5..29395d8 100755 --- a/Helper/CoreMarkdown.php +++ b/Helper/CoreMarkdown.php @@ -180,7 +180,7 @@ private function nummerizeCheckboxes(&$markup) $counter = new Counter; $count = 0; - $markup = preg_replace_callback("/activecheckbox\"/m", $counter->count(...), $markup, -1, $count, PREG_OFFSET_CAPTURE); + $markup = preg_replace_callback("/activecheckbox\"/m", array($counter, 'count'), $markup, -1, $count, PREG_OFFSET_CAPTURE); } } From aeeed40ac4ff0aeae520cdbe2e035eac1d112c28 Mon Sep 17 00:00:00 2001 From: Tomas Dittmann Date: Fri, 3 Nov 2023 11:58:55 +0100 Subject: [PATCH 3/7] use static counter --- Helper/CoreMarkdown.php | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Helper/CoreMarkdown.php b/Helper/CoreMarkdown.php index 29395d8..d36e24a 100755 --- a/Helper/CoreMarkdown.php +++ b/Helper/CoreMarkdown.php @@ -186,10 +186,10 @@ private function nummerizeCheckboxes(&$markup) class Counter { - private $count = 0; + private static $count = 0; public function count($matches) { - $this->count++; - return $matches[0][0] . " data-number=". $this->count . " "; + $this::$count++; + return $matches[0][0] . " data-number=". $this::$count . " "; } }; \ No newline at end of file From 52c7f4869dff7fffadcdace9151ecfc67ffbd4bf Mon Sep 17 00:00:00 2001 From: Tomas Dittmann Date: Wed, 22 Nov 2023 12:07:35 +0100 Subject: [PATCH 4/7] remove DOM-validation it checks agains an external DTD that will ratelimit too many requests. LoadHTML() itself already does everything we need --- vendor/erusev/parsedown-extra/ParsedownExtra.php | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/vendor/erusev/parsedown-extra/ParsedownExtra.php b/vendor/erusev/parsedown-extra/ParsedownExtra.php index 3cdea88..3df2f4f 100644 --- a/vendor/erusev/parsedown-extra/ParsedownExtra.php +++ b/vendor/erusev/parsedown-extra/ParsedownExtra.php @@ -628,11 +628,12 @@ protected function processTag($elementMarkup) # recursive $elementMarkup = mb_convert_encoding($elementMarkup, 'HTML-ENTITIES', 'UTF-8'); # http://stackoverflow.com/q/4879946/200145 - $DOMDocument->loadHTML($elementMarkup); - - if (!$DOMDocument->validate()) { - $errormessage = 'could not parse html
'; - $errors = libxml_get_errors(); + $DOMDocument->loadHTML($elementMarkup, LIBXML_DTDATTR | LIBXML_NONET); + + $errors = libxml_get_errors(); + if ($errors) + { + $errormessage = ""; foreach ($errors as $error) { $errormessage .= $error->message; $errormessage .= '
'; From 085539d86b0f69b1e11aec95093866e28ee0e836 Mon Sep 17 00:00:00 2001 From: Tomas Dittmann Date: Wed, 22 Nov 2023 12:10:13 +0100 Subject: [PATCH 5/7] add a title --- vendor/erusev/parsedown-extra/ParsedownExtra.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vendor/erusev/parsedown-extra/ParsedownExtra.php b/vendor/erusev/parsedown-extra/ParsedownExtra.php index 3df2f4f..2a09da4 100644 --- a/vendor/erusev/parsedown-extra/ParsedownExtra.php +++ b/vendor/erusev/parsedown-extra/ParsedownExtra.php @@ -633,7 +633,7 @@ protected function processTag($elementMarkup) # recursive $errors = libxml_get_errors(); if ($errors) { - $errormessage = ""; + $errormessage = "

htmlparser error:


"; foreach ($errors as $error) { $errormessage .= $error->message; $errormessage .= '
'; From 01627bf162d7ceefa23278871b6ea9d8515b3b97 Mon Sep 17 00:00:00 2001 From: Tomas Dittmann Date: Wed, 22 Nov 2023 13:55:33 +0100 Subject: [PATCH 6/7] further improve errorhandling add more output to help solving the problem --- vendor/erusev/parsedown-extra/ParsedownExtra.php | 16 ++++++++++++---- vendor/erusev/parsedown/Parsedown.php | 2 +- 2 files changed, 13 insertions(+), 5 deletions(-) diff --git a/vendor/erusev/parsedown-extra/ParsedownExtra.php b/vendor/erusev/parsedown-extra/ParsedownExtra.php index 2a09da4..efed016 100644 --- a/vendor/erusev/parsedown-extra/ParsedownExtra.php +++ b/vendor/erusev/parsedown-extra/ParsedownExtra.php @@ -628,20 +628,28 @@ protected function processTag($elementMarkup) # recursive $elementMarkup = mb_convert_encoding($elementMarkup, 'HTML-ENTITIES', 'UTF-8'); # http://stackoverflow.com/q/4879946/200145 - $DOMDocument->loadHTML($elementMarkup, LIBXML_DTDATTR | LIBXML_NONET); + $success = $DOMDocument->loadHTML($elementMarkup); + $DOMDocument->removeChild($DOMDocument->doctype); $errors = libxml_get_errors(); if ($errors) { - $errormessage = "

htmlparser error:


"; + $errormessage = "

HTML-parser error:


"; foreach ($errors as $error) { - $errormessage .= $error->message; + $errormessage .= "Line: " . $error->line . " - " . $error->message; $errormessage .= '
'; } return $errormessage; } - $DOMDocument->removeChild($DOMDocument->doctype); + if (!isset($DOMDocument->firstChild->firstChild->firstChild)) + { + $errormessage = "

General HTML-parser error:


"; + $errormessage .= "Input:
" . htmlspecialchars($elementMarkup) . "
"; + $errormessage .= "Output:
" . htmlspecialchars($DOMDocument->saveHTML()) . "
"; + return $errormessage; + } + $DOMDocument->replaceChild($DOMDocument->firstChild->firstChild->firstChild, $DOMDocument->firstChild); $elementText = ''; diff --git a/vendor/erusev/parsedown/Parsedown.php b/vendor/erusev/parsedown/Parsedown.php index 1832db7..01c2d6e 100644 --- a/vendor/erusev/parsedown/Parsedown.php +++ b/vendor/erusev/parsedown/Parsedown.php @@ -1977,7 +1977,7 @@ static function instance($name = 'default') protected $regexHtmlAttribute = '[a-zA-Z_:][\w:.-]*+(?:\s*+=\s*+(?:[^"\'=<>`\s]+|"[^"]*+"|\'[^\']*+\'))?+'; protected $voidElements = array( - 'area', 'base', 'br', 'col', 'command', 'embed', 'hr', 'img', 'input', 'link', 'meta', 'param', 'source', + 'area', 'base', 'br', 'col', 'command', 'embed', 'hr', 'img', 'input', 'link', 'meta', 'param', 'source', 'frame' ); protected $textLevelElements = array( From d896ca8d406a754309dc6a8d7b6769b7fd59df7a Mon Sep 17 00:00:00 2001 From: Tomas Dittmann Date: Wed, 22 Nov 2023 14:03:06 +0100 Subject: [PATCH 7/7] hint to a missing markdown=1 attribute revert adding "frame" to void-elements --- vendor/erusev/parsedown-extra/ParsedownExtra.php | 2 ++ vendor/erusev/parsedown/Parsedown.php | 2 +- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/vendor/erusev/parsedown-extra/ParsedownExtra.php b/vendor/erusev/parsedown-extra/ParsedownExtra.php index efed016..7a58c39 100644 --- a/vendor/erusev/parsedown-extra/ParsedownExtra.php +++ b/vendor/erusev/parsedown-extra/ParsedownExtra.php @@ -635,6 +635,7 @@ protected function processTag($elementMarkup) # recursive if ($errors) { $errormessage = "

HTML-parser error:


"; + $errormessage .= "The input was interpret as HTML - did you miss the markdown=1 attribute?
"; foreach ($errors as $error) { $errormessage .= "Line: " . $error->line . " - " . $error->message; $errormessage .= '
'; @@ -645,6 +646,7 @@ protected function processTag($elementMarkup) # recursive if (!isset($DOMDocument->firstChild->firstChild->firstChild)) { $errormessage = "

General HTML-parser error:


"; + $errormessage .= "The input was interpret as HTML - did you miss the markdown=1 attribute?
"; $errormessage .= "Input:
" . htmlspecialchars($elementMarkup) . "
"; $errormessage .= "Output:
" . htmlspecialchars($DOMDocument->saveHTML()) . "
"; return $errormessage; diff --git a/vendor/erusev/parsedown/Parsedown.php b/vendor/erusev/parsedown/Parsedown.php index 01c2d6e..d8b0f89 100644 --- a/vendor/erusev/parsedown/Parsedown.php +++ b/vendor/erusev/parsedown/Parsedown.php @@ -1977,7 +1977,7 @@ static function instance($name = 'default') protected $regexHtmlAttribute = '[a-zA-Z_:][\w:.-]*+(?:\s*+=\s*+(?:[^"\'=<>`\s]+|"[^"]*+"|\'[^\']*+\'))?+'; protected $voidElements = array( - 'area', 'base', 'br', 'col', 'command', 'embed', 'hr', 'img', 'input', 'link', 'meta', 'param', 'source', 'frame' + 'area', 'base', 'br', 'col', 'command', 'embed', 'hr', 'img', 'input', 'link', 'meta', 'param', 'source' ); protected $textLevelElements = array(