css_parser_1.1.0 1.1.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,6 @@
1
+ body {
2
+ color: black;
3
+ background: white;
4
+ }
5
+
6
+ p { margin: 0px; }
@@ -0,0 +1,3 @@
1
+ @import "../simple.css";
2
+
3
+ a { text-decoration: none; }
@@ -0,0 +1,56 @@
1
+ require File.dirname(__FILE__) + '/test_helper'
2
+
3
+ # Test cases for reading and generating CSS shorthand properties
4
+ class CssParserBasicTests < Test::Unit::TestCase
5
+ include CssParser
6
+
7
+ def setup
8
+ @cp = CssParser::Parser.new
9
+ @css = <<-EOT
10
+ html, body, p { margin: 0px; }
11
+ p { padding: 0px; }
12
+ #content { font: 12px/normal sans-serif; }
13
+ EOT
14
+ end
15
+
16
+ def test_finding_by_selector
17
+ @cp.add_block!(@css)
18
+ assert_equal 'margin: 0px;', @cp.find_by_selector('body').join(' ')
19
+ assert_equal 'margin: 0px; padding: 0px;', @cp.find_by_selector('p').join(' ')
20
+ end
21
+
22
+ def test_adding_block
23
+ @cp.add_block!(@css)
24
+ assert_equal 'margin: 0px;', @cp.find_by_selector('body').join
25
+ end
26
+
27
+ def test_adding_a_rule
28
+ @cp.add_rule!('div', 'color: blue;')
29
+ assert_equal 'color: blue;', @cp.find_by_selector('div').join(' ')
30
+ end
31
+
32
+ def test_adding_a_rule_set
33
+ rs = CssParser::RuleSet.new('div', 'color: blue;')
34
+ @cp.add_rule_set!(rs)
35
+ assert_equal 'color: blue;', @cp.find_by_selector('div').join(' ')
36
+ end
37
+
38
+ def test_toggling_uri_conversion
39
+ # with conversion
40
+ cp_with_conversion = Parser.new(:absolute_paths => true)
41
+ cp_with_conversion.add_block!("body { background: url('../style/yellow.png?abc=123') };",
42
+ :base_uri => 'http://example.org/style/basic.css')
43
+
44
+ assert_equal "background: url('http://example.org/style/yellow.png?abc=123');",
45
+ cp_with_conversion['body'].join(' ')
46
+
47
+ # without conversion
48
+ cp_without_conversion = Parser.new(:absolute_paths => false)
49
+ cp_without_conversion.add_block!("body { background: url('../style/yellow.png?abc=123') };",
50
+ :base_uri => 'http://example.org/style/basic.css')
51
+
52
+ assert_equal "background: url('../style/yellow.png?abc=123');",
53
+ cp_without_conversion['body'].join(' ')
54
+ end
55
+
56
+ end
@@ -0,0 +1,110 @@
1
+ require File.dirname(__FILE__) + '/test_helper'
2
+
3
+ # Test cases for the CssParser's loading functions.
4
+ class CssParserLoadingTests < Test::Unit::TestCase
5
+ include CssParser
6
+ include WEBrick
7
+
8
+ def setup
9
+ # from http://nullref.se/blog/2006/5/17/testing-with-webrick
10
+ @cp = Parser.new
11
+
12
+ @uri_base = 'http://localhost:12000'
13
+
14
+ www_root = File.dirname(__FILE__) + '/fixtures/'
15
+
16
+ @server_thread = Thread.new do
17
+ s = WEBrick::HTTPServer.new(:Port => 12000, :DocumentRoot => www_root, :Logger => Log.new(nil, BasicLog::FATAL), :AccessLog => [])
18
+ @port = s.config[:Port]
19
+ begin
20
+ s.start
21
+ ensure
22
+ s.shutdown
23
+ end
24
+ end
25
+
26
+ sleep 1 # ensure the server has time to load
27
+ end
28
+
29
+ def teardown
30
+ @server_thread.kill
31
+ @server_thread.join(5)
32
+ @server_thread = nil
33
+ end
34
+
35
+ def test_loading_a_local_file
36
+ file_name = File.dirname(__FILE__) + '/fixtures/simple.css'
37
+ @cp.load_file!(file_name)
38
+ assert_equal 'margin: 0px;', @cp.find_by_selector('p').join(' ')
39
+ end
40
+
41
+ def test_loading_a_remote_file
42
+ @cp.load_uri!("#{@uri_base}/simple.css")
43
+ assert_equal 'margin: 0px;', @cp.find_by_selector('p').join(' ')
44
+ end
45
+
46
+ def test_following_at_import_rules_local
47
+ base_dir = File.dirname(__FILE__) + '/fixtures'
48
+ @cp.load_file!('import1.css', base_dir)
49
+
50
+ # from '/import1.css'
51
+ assert_equal 'color: lime;', @cp.find_by_selector('div').join(' ')
52
+
53
+ # from '/subdir/import2.css'
54
+ assert_equal 'text-decoration: none;', @cp.find_by_selector('a').join(' ')
55
+
56
+ # from '/subdir/../simple.css'
57
+ assert_equal 'margin: 0px;', @cp.find_by_selector('p').join(' ')
58
+ end
59
+
60
+ def test_following_at_import_rules_remote
61
+ @cp.load_uri!("#{@uri_base}/import1.css")
62
+
63
+ # from '/import1.css'
64
+ assert_equal 'color: lime;', @cp.find_by_selector('div').join(' ')
65
+
66
+ # from '/subdir/import2.css'
67
+ assert_equal 'text-decoration: none;', @cp.find_by_selector('a').join(' ')
68
+
69
+ # from '/subdir/../simple.css'
70
+ assert_equal 'margin: 0px;', @cp.find_by_selector('p').join(' ')
71
+ end
72
+
73
+ def test_following_at_import_rules_from_add_block
74
+ css_block = '@import "../simple.css";'
75
+
76
+ @cp.add_block!(css_block, :base_uri => "#{@uri_base}/subdir/")
77
+
78
+ # from 'simple.css'
79
+ assert_equal 'margin: 0px;', @cp.find_by_selector('p').join(' ')
80
+ end
81
+
82
+ def test_importing_with_media_types
83
+ @cp.load_uri!("#{@uri_base}/import-with-media-types.css")
84
+
85
+ # from simple.css with :screen media type
86
+ assert_equal 'margin: 0px;', @cp.find_by_selector('p', :screen).join(' ')
87
+ assert_equal '', @cp.find_by_selector('p', :tty).join(' ')
88
+ end
89
+
90
+ def test_throwing_circular_reference_exception
91
+ assert_raise CircularReferenceError do
92
+ @cp.load_uri!("#{@uri_base}/import-circular-reference.css")
93
+ end
94
+ end
95
+
96
+ def test_toggling_not_found_exceptions
97
+ cp_with_exceptions = Parser.new(:io_exceptions => true)
98
+
99
+ assert_raise RemoteFileError do
100
+ cp_with_exceptions.load_uri!("#{@uri_base}/no-exist.xyz")
101
+ end
102
+
103
+ cp_without_exceptions = Parser.new(:io_exceptions => false)
104
+
105
+ assert_nothing_raised RemoteFileError do
106
+ cp_without_exceptions.load_uri!("#{@uri_base}/no-exist.xyz")
107
+ end
108
+ end
109
+
110
+ end
@@ -0,0 +1,71 @@
1
+ require File.dirname(__FILE__) + '/test_helper'
2
+
3
+ # Test cases for the handling of media types
4
+ class CssParserMediaTypesTests < Test::Unit::TestCase
5
+ include CssParser
6
+
7
+ def setup
8
+ @cp = Parser.new
9
+ end
10
+
11
+ def test_finding_by_media_type
12
+ # from http://www.w3.org/TR/CSS21/media.html#at-media-rule
13
+ css = <<-EOT
14
+ @media print {
15
+ body { font-size: 10pt }
16
+ }
17
+ @media screen {
18
+ body { font-size: 13px }
19
+ }
20
+ @media screen, print {
21
+ body { line-height: 1.2 }
22
+ }
23
+ EOT
24
+
25
+ @cp.add_block!(css)
26
+
27
+ assert_equal 'font-size: 10pt; line-height: 1.2;', @cp.find_by_selector('body', :print).join(' ')
28
+ assert_equal 'font-size: 13px; line-height: 1.2;', @cp.find_by_selector('body', :screen).join(' ')
29
+ end
30
+
31
+ def atest_finding_by_multiple_media_types
32
+ css = <<-EOT
33
+ @media print {
34
+ body { font-size: 10pt }
35
+ }
36
+ @media handheld {
37
+ body { font-size: 13px }
38
+ }
39
+ @media screen, print {
40
+ body { line-height: 1.2 }
41
+ }
42
+ EOT
43
+ @cp.add_block!(css)
44
+
45
+ assert_equal 'font-size: 13px; line-height: 1.2;', @cp.find_by_selector('body', [:screen,:handheld]).join(' ')
46
+ end
47
+
48
+ def test_adding_block_with_media_types
49
+ css = <<-EOT
50
+ body { font-size: 10pt }
51
+ EOT
52
+
53
+ @cp.add_block!(css, :media_types => [:screen])
54
+
55
+ assert_equal 'font-size: 10pt;', @cp.find_by_selector('body', :screen).join(' ')
56
+ assert @cp.find_by_selector('body', :handheld).empty?
57
+ end
58
+
59
+ def atest_adding_rule_set_with_media_type
60
+ @cp.add_rule!('body', 'color: black;', [:handheld,:tty])
61
+ @cp.add_rule!('body', 'color: blue;', :screen)
62
+ assert_equal 'color: black;', @cp.find_by_selector('body', :handheld).join(' ')
63
+ end
64
+
65
+ def atest_selecting_with_all_meda_type
66
+ @cp.add_rule!('body', 'color: black;', [:handheld,:tty])
67
+ assert_equal 'color: black;', @cp.find_by_selector('body', :all).join(' ')
68
+ end
69
+
70
+
71
+ end
@@ -0,0 +1,143 @@
1
+ require File.dirname(__FILE__) + '/test_helper'
2
+
3
+ # Test cases for the CssParser.
4
+ class CssParserTests < Test::Unit::TestCase
5
+ include CssParser
6
+
7
+ def setup
8
+ @cp = Parser.new
9
+ end
10
+
11
+ def test_at_page_rule
12
+ # from http://www.w3.org/TR/CSS21/page.html#page-selectors
13
+ css = <<-EOT
14
+ @page { margin: 2cm }
15
+
16
+ @page :first {
17
+ margin-top: 10cm
18
+ }
19
+ EOT
20
+
21
+ @cp.add_block!(css)
22
+
23
+ assert_equal 'margin: 2cm;', @cp.find_by_selector('@page').join(' ')
24
+ assert_equal 'margin-top: 10cm;', @cp.find_by_selector('@page :first').join(' ')
25
+ end
26
+
27
+ def test_should_ignore_comments
28
+ # see http://www.w3.org/Style/CSS/Test/CSS2.1/current/html4/t040109-c17-comments-00-b.htm
29
+ css =<<-EOT
30
+ /* This is a CSS comment. */
31
+ .one {color: green;} /* Another comment */
32
+ /* The following should not be used:
33
+ .one {color: red;} */
34
+ .two {color: green; /* color: yellow; */}
35
+ /**
36
+ .three {color: red;} */
37
+ .three {color: green;}
38
+ /**/
39
+ .four {color: green;}
40
+ /*********/
41
+ .five {color: green;}
42
+ /* a comment **/
43
+ .six {color: green;}
44
+ EOT
45
+
46
+ @cp.add_block!(css)
47
+ @cp.each_selector do |sel, decs, spec|
48
+ assert_equal 'color: green;', decs
49
+ end
50
+ end
51
+
52
+ def test_parsing_blocks
53
+ # dervived from http://www.w3.org/TR/CSS21/syndata.html#rule-sets
54
+ css = <<-EOT
55
+ div[name='test'] {
56
+
57
+ color:
58
+
59
+ red;
60
+
61
+ }div:hover{coloR:red;
62
+ }div:first-letter{color:red;/*color:blue;}"commented out"*/}
63
+
64
+ p[example="public class foo\
65
+ {\
66
+ private string x;\
67
+ \
68
+ foo(int x) {\
69
+ this.x = 'test';\
70
+ this.x = \"test\";\
71
+ }\
72
+ \
73
+ }"] { color: red }
74
+
75
+ p { color:red}
76
+ EOT
77
+
78
+ @cp.add_block!(css)
79
+
80
+ @cp.each_selector do |sel, decs, spec|
81
+ assert_equal 'color: red;', decs
82
+ end
83
+ end
84
+
85
+ def test_ignoring_malformed_declarations
86
+ flunk
87
+ # dervived from http://www.w3.org/TR/CSS21/syndata.html#parsing-errors
88
+ css = <<-EOT
89
+ p { color:green }
90
+ p { color:green; color } /* malformed declaration missing ':', value */
91
+ p { color:red; color; color:green } /* same with expected recovery */
92
+ p { color:green; color: } /* malformed declaration missing value */
93
+ p { color:red; color:; color:green } /* same with expected recovery */
94
+ p { color:green; color{;color:maroon} } /* unexpected tokens { } */
95
+ p { color:red; color{;color:maroon}; color:green } /* same with recovery */
96
+ EOT
97
+
98
+ @cp.add_block!(css)
99
+
100
+ @cp.each_selector do |sel, decs, spec|
101
+ assert_equal 'color: green;', decs
102
+ end
103
+ end
104
+
105
+ def test_calculating_specificity
106
+ # from http://www.w3.org/TR/CSS21/cascade.html#specificity
107
+ assert_equal 0, CssParser.calculate_specificity('*')
108
+ assert_equal 1, CssParser.calculate_specificity('li')
109
+ assert_equal 2, CssParser.calculate_specificity('li:first-line')
110
+ assert_equal 2, CssParser.calculate_specificity('ul li')
111
+ assert_equal 3, CssParser.calculate_specificity('ul ol+li')
112
+ assert_equal 11, CssParser.calculate_specificity('h1 + *[rel=up]')
113
+ assert_equal 13, CssParser.calculate_specificity('ul ol li.red')
114
+ assert_equal 21, CssParser.calculate_specificity('li.red.level')
115
+ assert_equal 100, CssParser.calculate_specificity('#x34y')
116
+
117
+ # from http://www.hixie.ch/tests/adhoc/css/cascade/specificity/003.html
118
+ assert_equal CssParser.calculate_specificity('div *'), CssParser.calculate_specificity('p')
119
+ assert CssParser.calculate_specificity('body div *') > CssParser.calculate_specificity('div *')
120
+
121
+ # other tests
122
+ assert_equal 11, CssParser.calculate_specificity('h1[id|=123]')
123
+ end
124
+
125
+ def test_converting_uris
126
+ base_uri = 'http://www.example.org/style/basic.css'
127
+ ["body { background: url(yellow) };", "body { background: url('yellow') };",
128
+ "body { background: url('/style/yellow') };",
129
+ "body { background: url(\"../style/yellow\") };",
130
+ "body { background: url(\"lib/../../style/yellow\") };"].each do |css|
131
+ converted_css = CssParser.convert_uris(css, base_uri)
132
+ assert_equal "body { background: url('http://www.example.org/style/yellow') };", converted_css
133
+ end
134
+
135
+ converted_css = CssParser.convert_uris("body { background: url(../style/yellow-dot_symbol$.png?abc=123&amp;def=456&ghi=789#1011) };", base_uri)
136
+ assert_equal "body { background: url('http://www.example.org/style/yellow-dot_symbol$.png?abc=123&amp;def=456&ghi=789#1011') };", converted_css
137
+
138
+ # taken from error log: 2007-10-23 04:37:41#2399
139
+ converted_css = CssParser.convert_uris('.specs {font-family:Helvetica;font-weight:bold;font-style:italic;color:#008CA8;font-size:1.4em;list-style-image:url("images/bullet.gif");}', 'http://www.example.org/directory/file.html')
140
+ assert_equal ".specs {font-family:Helvetica;font-weight:bold;font-style:italic;color:#008CA8;font-size:1.4em;list-style-image:url('http://www.example.org/directory/images/bullet.gif');}", converted_css
141
+ end
142
+
143
+ end
@@ -0,0 +1,69 @@
1
+ # coding: iso-8859-1
2
+ require File.dirname(__FILE__) + '/test_helper'
3
+
4
+ # Test cases for CSS regular expressions
5
+ #
6
+ # see http://www.w3.org/TR/CSS21/syndata.html and
7
+ # http://www.w3.org/TR/CSS21/grammar.html
8
+ class CssParserRegexpTests < Test::Unit::TestCase
9
+ def test_strings
10
+ # complete matches
11
+ ['"abcd"', '" A sd s�drcv \'dsf\' asd rfg asd"', '"A\ d??ef 123!"',
12
+ "\"this is\\\n a test\"", '"back\67round"', '"r\000065 ed"',
13
+ "'abcd'", "' A sd sedrcv \"dsf\" asd rf�&23$%#%$g asd'", "'A\\\n def 123!'",
14
+ "'this is\\\n a test'", "'back\\67round'", "'r\\000065 ed'"
15
+ ].each do |str|
16
+ assert_equal str, str.match(CssParser::RE_STRING).to_s
17
+ end
18
+
19
+ test_string = "p { background: red url(\"url\\.'p'ng\"); }"
20
+ assert_equal "\"url\\.'p'ng\"", test_string.match(CssParser::RE_STRING).to_s
21
+
22
+ end
23
+
24
+ def test_unicode
25
+ ['back\67round', 'r\000065 ed', '\00006C'].each do |str|
26
+ assert_match(Regexp.new(CssParser::RE_UNICODE), str)
27
+ end
28
+ end
29
+
30
+ def test_colour
31
+ ['color: #fff', 'color:#f0a09c;', 'color: #04A', 'color: #04a9CE',
32
+ 'color: rgb(100, -10%, 300);', 'color: rgb(10,10,10)', 'color:rgb(12.7253%, -12%,0)',
33
+ 'color: black', 'color:Red;', 'color: AqUa;', 'color: blue ', 'color: transparent'
34
+ ].each do |colour|
35
+ assert_match(CssParser::RE_COLOUR, colour)
36
+ end
37
+
38
+ ['color: #fa', 'color:#f009c;', 'color: #04G', 'color: #04a9Cq',
39
+ 'color: rgb 100, -10%, 300;', 'color: rgb 10,10,10', 'color:rgb(12px, -12%,0)',
40
+ 'color:fuscia;', 'color: thick'
41
+ ].each do |colour|
42
+ assert_no_match(CssParser::RE_COLOUR, colour)
43
+ end
44
+ end
45
+
46
+ def test_uris
47
+ crazy_uri = 'http://www.example.com:80/~/redb%20all.png?test=test&test;test+test#test!'
48
+
49
+ assert_equal "url('#{crazy_uri}')",
50
+ "li { list-style: url('#{crazy_uri}') disc }".match(CssParser::RE_URI).to_s
51
+
52
+ assert_equal "url(#{crazy_uri})",
53
+ "li { list-style: url(#{crazy_uri}) disc }".match(CssParser::RE_URI).to_s
54
+
55
+ assert_equal "url(\"#{crazy_uri}\")",
56
+ "li { list-style: url(\"#{crazy_uri}\") disc }".match(CssParser::RE_URI).to_s
57
+ end
58
+
59
+
60
+ protected
61
+ def load_test_file(filename)
62
+ fh = File.new("fixtures/#{filename}", 'r')
63
+ test_file = fh.read
64
+ fh.close
65
+
66
+ return test_file
67
+ end
68
+
69
+ end