-
Notifications
You must be signed in to change notification settings - Fork 62
/
Copy pathHTMLCustomGenerator.py
62 lines (47 loc) · 2.31 KB
/
HTMLCustomGenerator.py
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
# Copyright (c) 2017-2023 Renata Hodovan, Akos Kiss.
#
# Licensed under the BSD 3-Clause License
# <LICENSE.rst or https://opensource.org/licenses/BSD-3-Clause>.
# This file may not be copied, modified, or distributed except
# according to those terms.
import json
import random
from os.path import dirname, join
from grammarinator.runtime import UnlexerRule, UnparserRuleContext
from HTMLGenerator import HTMLGenerator
with open(join(dirname(__file__), 'html.json')) as f:
tags = json.load(f)
tag_names = list(tags.keys())
class HTMLCustomGenerator(HTMLGenerator):
attr_stack = []
tag_stack = []
tags = set()
# Customize the function generated from the htmlTagName parser rule to produce valid tag names.
def htmlTagName(self, parent=None):
with UnparserRuleContext(gen=self, name='htmlTagName', parent=parent) as rule:
current = rule.current
name = random.choice(tags[self.tag_stack[-1]]['children'] or tag_names if self.tag_stack else tag_names)
self.tag_stack.append(name)
current += UnlexerRule(src=name)
self.tag_stack.append(name)
return current
# Customize the function generated from the htmlAttributeName parser rule to produce valid attribute names.
def htmlAttributeName(self, parent=None):
with UnparserRuleContext(gen=self, name='htmlAttributeName', parent=parent) as rule:
current = rule.current
name = random.choice(list(tags[self.tag_stack[-1]]['attributes'].keys()) or ['""'])
self.attr_stack.append(name)
current += UnlexerRule(src=name)
return current
# Customize the function generated from the htmlAttributeValue parser rule to produce valid attribute values
# to the current tag and attribute name.
def htmlAttributeValue(self, parent=None):
with UnparserRuleContext(gen=self, name='htmlAttributeValue', parent=parent) as rule:
current = rule.current
current += UnlexerRule(src=random.choice(tags[self.tag_stack[-1]]['attributes'].get(self.attr_stack.pop(), ['""']) or ['""']))
return current
def _endOfHtmlElement(self):
self.tag_stack.pop()
# You probably want to rewrite this with a distinct CSS fuzzer.
def _style_sheet(self):
return '* { background: green; }'