-
Notifications
You must be signed in to change notification settings - Fork 157
/
Copy pathHtmlBindingSniff.php
92 lines (85 loc) · 2.75 KB
/
HtmlBindingSniff.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
<?php
/**
* Copyright 2021 Adobe
* All Rights Reserved.
*/
declare(strict_types=1);
namespace Magento2\Sniffs\Html;
use PHP_CodeSniffer\Sniffs\Sniff;
use PHP_CodeSniffer\Files\File;
/**
* Sniffing improper HTML bindings.
*/
class HtmlBindingSniff implements Sniff
{
/**
* @inheritDoc
*/
public function register()
{
return [T_INLINE_HTML];
}
/**
* Load HTML document to validate.
*
* @param int $stackPointer
* @param File $file
* @return \DOMDocument|null
*/
private function loadHtmlDocument(int $stackPointer, File $file): ?\DOMDocument
{
if ($stackPointer === 0) {
$html = $file->getTokensAsString($stackPointer, count($file->getTokens()));
$dom = new \DOMDocument();
try {
// phpcs:disable Generic.PHP.NoSilencedErrors
@$dom->loadHTML($html);
return $dom;
} catch (\Throwable $exception) {
return null;
}
}
return null;
}
/**
* @inheritDoc
*
* Find HTML data bindings and check variables used.
*/
public function process(File $phpcsFile, $stackPtr)
{
if (!$dom = $this->loadHtmlDocument($stackPtr, $phpcsFile)) {
return;
}
/** @var string[] $htmlBindings */
$htmlBindings = [];
$domXpath = new \DOMXPath($dom);
$dataBindAttributes = $domXpath->query('//@*[name() = "data-bind"]');
foreach ($dataBindAttributes as $dataBindAttribute) {
$knockoutBinding = $dataBindAttribute->nodeValue;
preg_match('/^(.+\s*?)?html\s*?\:(.+)/ims', $knockoutBinding, $htmlBindingStart);
if ($htmlBindingStart) {
$htmlBinding = trim(preg_replace('/\,[a-z0-9\_\s]+\:.+/ims', '', $htmlBindingStart[2]));
$htmlBindings[] = $htmlBinding;
}
}
$htmlAttributes = $domXpath->query('//@*[name() = "html"]');
foreach ($htmlAttributes as $htmlAttribute) {
$magentoBinding = $htmlAttribute->nodeValue;
$htmlBindings[] = trim($magentoBinding);
}
foreach ($htmlBindings as $htmlBinding) {
if (!preg_match('/^[0-9\\\'\"]/ims', $htmlBinding)
&& !preg_match('/UnsanitizedHtml(\(.*?\))*?$/', $htmlBinding)
) {
$phpcsFile->addError(
'Variables/functions used for HTML binding must have UnsanitizedHtml suffix'
. ' - "' . $htmlBinding . '" doesn\'t,' . PHP_EOL
. 'consider using text binding if the value is supposed to be text',
null,
'KnockoutBindingHtmlSuffix'
);
}
}
}
}