1#!/usr/bin/python3
2#
3# Copyright (c) 2018-2019 Collabora, Ltd.
4#
5# SPDX-License-Identifier: Apache-2.0
6#
7# Author(s):    Ryan Pavlik <ryan.pavlik@collabora.com>
8#
9# Purpose:      This file contains tests for check_spec_links.py
10
11import pytest
12
13from check_spec_links import MessageId, makeMacroChecker
14from spec_tools.console_printer import ConsolePrinter
15from spec_tools.macro_checker_file import shouldEntityBeText
16
17# API-specific constants
18PROTO = 'vkCreateInstance'
19STRUCT = 'VkInstanceCreateInfo'
20EXT = 'VK_KHR_display'
21
22
23class CheckerWrapper(object):
24    """Little wrapper object for a MacroChecker.
25
26    Intended for use in making test assertions shorter and easier to read."""
27
28    def __init__(self, capsys):
29        self.ckr = makeMacroChecker(set())
30        self.capsys = capsys
31
32    def enabled(self, enabled_messages):
33        """Updates the checker's enable message type set, from an iterable."""
34        self.ckr.enabled_messages = set(enabled_messages)
35        return self
36
37    def check(self, string):
38        """Checks a string (as if it were a file), outputs the results to the console, then returns the MacroCheckerFile."""
39
40        # Flush the captured output.
41        _ = self.capsys.readouterr()
42
43        # Process
44        f = self.ckr.processString(string + '\n')
45
46        # Dump messages
47        ConsolePrinter().output(f)
48        return f
49
50
51@pytest.fixture
52def ckr(capsys):
53    """Fixture - add an arg named ckr to your test function to automatically get one passed to you."""
54    return CheckerWrapper(capsys)
55
56
57def msgReplacement(file_checker, which=0):
58    """Return the replacement text associated with the specified message."""
59    assert(len(file_checker.messages) > which)
60    msg = file_checker.messages[which]
61    from pprint import pprint
62    pprint(msg.script_location)
63    pprint(msg.replacement)
64    pprint(msg.fix)
65    return msg.replacement
66
67
68def loneMsgReplacement(file_checker):
69    """Assert there's only one message in a file checker, and return the replacement text associated with it."""
70    assert(len(file_checker.messages) == 1)
71    return msgReplacement(file_checker)
72
73
74def message(file_checker, which=0):
75    """Return a string of the message lines associated with the message of a file checker."""
76    assert(len(file_checker.messages) > which)
77    return "\n".join(file_checker.messages[which].message)
78
79
80def allMessages(file_checker):
81    """Return a list of strings, each being the combination of the message lines of a message from a file checker."""
82    return ['\n'.join(msg.message) for msg in file_checker.messages]
83
84
85def test_missing_macro(ckr):
86    """Verify correct functioning of MessageId.MISSING_MACRO."""
87    ckr.enabled([MessageId.MISSING_MACRO])
88
89    # This should have a missing macro warning
90    assert(ckr.check('with %s by' % PROTO).numDiagnostics() == 1)
91
92    # These 3 should not have a missing macro warning because of their context
93    # (in a link)
94    assert(not ckr.check('<<%s' % PROTO).messages)
95    # These 2 are simulating links that broke over lines
96    assert(not ckr.check('%s>>' % PROTO).messages)
97    assert(not ckr.check(
98        '%s asdf>> table' % PROTO).messages)
99
100
101def test_entity_detection(ckr):
102    ckr.enabled([MessageId.BAD_ENTITY])
103    # Should complain about BAD_ENTITY
104    assert(ckr.check('flink:abcd').numDiagnostics() == 1)
105
106    # Should just give BAD_ENTITY (an error), not MISSING_TEXT (a warning).
107    # Verifying that wrapping in asterisks (for formatting) doesn't get picked up as
108    # an asterisk in the entity name (a placeholder).
109    ckr.enabled(
110        [MessageId.MISSING_TEXT, MessageId.BAD_ENTITY])
111    assert(ckr.check('*flink:abcd*').numErrors() == 1)
112
113
114def test_wrong_macro(ckr):
115    ckr.enabled([MessageId.WRONG_MACRO])
116    # Should error - this ought to be code:uint32_t
117    assert(ckr.check('basetype:uint32_t').numErrors() == 1)
118
119    # This shouldn't error
120    assert(ckr.check('code:uint32_t').numErrors() == 0)
121
122
123def test_should_entity_be_text():
124    # These 5 are all examples of patterns that would merit usage of a ptext/etext/etc
125    # macro, for various reasons:
126
127    # has variable in subscript
128    assert(shouldEntityBeText('pBuffers[i]', '[i]'))
129    assert(shouldEntityBeText('API_ENUM_[X]', '[X]'))
130
131    # has asterisk
132    assert(shouldEntityBeText('maxPerStage*', None))
133
134    # double-underscores make italicized placeholders
135    # (triple are double-underscores delimited by underscores...)
136    assert(shouldEntityBeText('API_ENUM[__x__]', '[__x__]'))
137    assert(shouldEntityBeText('API_ENUM___i___EXT', None))
138
139    # This shouldn't be a *text: macro because it only has single underscores
140    assert(False == shouldEntityBeText('API_ENUM_i_EXT', None))
141
142
143def test_misused_text(ckr):
144    # Tests the same patterns as test_should_entity_be_text(),
145    # but in a whole checker
146    ckr.enabled([MessageId.MISUSED_TEXT])
147
148    assert(ckr.check('etext:API_ENUM_').numDiagnostics() == 0)
149    assert(ckr.check('etext:API_ENUM_[X]').numDiagnostics() == 0)
150    assert(ckr.check('etext:API_ENUM[i]').numDiagnostics() == 0)
151    assert(ckr.check('etext:API_ENUM[__x__]').numDiagnostics() == 0)
152
153    # Should be OK, since __i__ is a placeholder here
154    assert(ckr.check('etext:API_ENUM___i___EXT').numDiagnostics() == 0)
155
156    # This shouldn't be a *text: macro because it only has single underscores
157    assert(ckr.check('API_ENUM_i_EXT').numDiagnostics() == 0)
158
159
160def test_extension(ckr):
161    ckr.enabled(set(MessageId))
162    # Check formatting of extension names:
163    # the following is the canonical way to refer to an extension
164    # (link wrapped in backticks)
165    expected_replacement = '`<<%s>>`' % EXT
166
167    # Extension name mentioned without any markup, should be added
168    assert(loneMsgReplacement(ckr.check('asdf %s asdf' % EXT))
169           == expected_replacement)
170
171    # Extension name mentioned without any markup and wrong case,
172    # should be added and have case fixed
173    assert(loneMsgReplacement(ckr.check('asdf %s asdf' % EXT.upper()))
174           == expected_replacement)
175
176    # Extension name using wrong/old macro: ename isn't for extensions.
177    assert(loneMsgReplacement(ckr.check('asdf ename:%s asdf' % EXT))
178           == expected_replacement)
179
180    # Extension name using wrong macro: elink isn't for extensions.
181    assert(loneMsgReplacement(ckr.check('asdf elink:%s asdf' % EXT))
182           == expected_replacement)
183
184    # Extension name using wrong macro and wrong case: should have markup and
185    # case fixed
186    assert(loneMsgReplacement(ckr.check('asdf elink:%s asdf' % EXT.upper()))
187           == expected_replacement)
188
189    # This shouldn't cause errors because this is how we want it to look.
190    assert(not ckr.check('asdf `<<%s>>` asdf' % EXT).messages)
191
192    # This doesn't (shouldn't?) cause errors because just backticks on their own
193    # "escape" names from the "missing markup" tests.
194    assert(not ckr.check('asdf `%s` asdf' % EXT).messages)
195
196    # TODO can we auto-correct this to add the backticks?
197    # Doesn't error now, but would be nice if it did...
198    assert(not ckr.check('asdf <<%s>> asdf' % EXT).messages)
199
200
201def test_refpage_tag(ckr):
202    ckr.enabled([MessageId.REFPAGE_TAG])
203
204    # Should error: missing refpage='' field
205    assert(ckr.check("[open,desc='',type='',xrefs='']").numErrors() == 1)
206    # Should error: missing desc='' field
207    assert(ckr.check("[open,refpage='',type='',xrefs='']").numErrors() == 1)
208    # Should error: missing type='' field
209    assert(ckr.check("[open,refpage='',desc='',xrefs='']").numErrors() == 1)
210
211    # Should not error: missing xrefs field is optional
212    assert(not ckr.check("[open,refpage='',desc='',type='']").messages)
213
214    # Should error, due to missing refpage, but not crash due to message printing (note the unicode smart quote)
215    assert(ckr.check("[open,desc='',type='',xrefs=’']").numDiagnostics() == 1)
216
217
218def test_refpage_name(ckr):
219    ckr.enabled([MessageId.REFPAGE_NAME])
220    # Should not error: actually exists.
221    assert(ckr.check(
222        "[open,refpage='%s',desc='',type='']" % PROTO).numDiagnostics() == 0)
223
224    # Should error: does not exist.
225    assert(
226        ckr.check("[open,refpage='bogus',desc='',type='']").numDiagnostics() == 1)
227
228
229def test_refpage_missing_desc(ckr):
230    ckr.enabled([MessageId.REFPAGE_MISSING_DESC])
231    # Should not warn: non-empty description actually exists.
232    assert(ckr.check(
233        "[open,refpage='',desc='non-empty description',type='']").numDiagnostics() == 0)
234
235    # Should warn: desc field is empty.
236    assert(
237        ckr.check("[open,refpage='',desc='',type='']").numDiagnostics() == 1)
238
239
240def test_refpage_type(ckr):
241    ckr.enabled([MessageId.REFPAGE_TYPE])
242    # Should not error: this is of type 'protos'.
243    assert(not ckr.check(
244        "[open,refpage='%s',desc='',type='protos']" % PROTO).messages)
245
246    # Should error: this is of type 'protos', not 'structs'.
247    assert(
248        ckr.check("[open,refpage='%s',desc='',type='structs']" % PROTO).messages)
249
250
251def test_refpage_xrefs(ckr):
252    ckr.enabled([MessageId.REFPAGE_XREFS])
253    # Should not error: this is a valid entity to have an xref to.
254    assert(not ckr.check(
255        "[open,refpage='',desc='',type='protos',xrefs='%s']" % STRUCT).messages)
256
257    # case difference:
258    # should error but offer a replacement.
259    assert(loneMsgReplacement(ckr.check("[open,refpage='',xrefs='%s']" % STRUCT.lower()))
260           == STRUCT)
261
262    # Should error: not a valid entity.
263    assert(ckr.check(
264        "[open,refpage='',desc='',type='protos',xrefs='bogus']").numDiagnostics() == 1)
265
266
267def test_refpage_xrefs_comma(ckr):
268    ckr.enabled([MessageId.REFPAGE_XREFS_COMMA])
269    # Should not error: no commas in the xrefs field
270    assert(not ckr.check(
271        "[open,refpage='',xrefs='abc']").messages)
272
273    # Should error: commas shouldn't be there since it's space-delimited.
274    assert(loneMsgReplacement(
275        ckr.check("[open,refpage='',xrefs='abc,']")) == 'abc')
276
277    # All should correct to the same thing.
278    equivalent_tags_with_commas = [
279        "[open,refpage='',xrefs='abc, 123']",
280        "[open,refpage='',xrefs='abc,123']",
281        "[open,refpage='',xrefs='abc , 123']"]
282    for has_comma in equivalent_tags_with_commas:
283        assert(loneMsgReplacement(ckr.check(has_comma)) == 'abc 123')
284
285
286def test_refpage_block(ckr):
287    """Tests of the REFPAGE_BLOCK message."""
288    ckr.enabled([MessageId.REFPAGE_BLOCK])
289    # Should not error: have the tag, an open, and a close
290    assert(not ckr.check(
291        """[open,]
292        --
293        bla
294        --""").messages)
295    assert(not ckr.check(
296        """[open,refpage='abc']
297        --
298        bla
299        --
300
301        [open,refpage='123']
302        --
303        bla2
304        --""").messages)
305
306    # Should have 1 error: file ends immediately after tag
307    assert(ckr.check(
308        "[open,]").numDiagnostics() == 1)
309
310    # Should have 1 error: line after tag isn't --
311    assert(ckr.check(
312        """[open,]
313        bla
314        --""").numDiagnostics() == 1)
315    # Checking precedence of checks: this should have 1 error because line after tag isn't --
316    # (but it is something that causes a line to be handled differently)
317    assert(ckr.check(
318        """[open,]
319        == Heading
320        --""").numDiagnostics() == 1)
321    assert(ckr.check(
322        """[open,]
323        ----
324        this is in a code block
325        ----
326        --""").numDiagnostics() == 1)
327
328    # Should have 1 error: tag inside refpage.
329    tag_inside = """[open,]
330        --
331        bla
332        [open,]
333        --"""
334    assert(ckr.check(tag_inside).numDiagnostics() == 1)
335    assert("already in a refpage block" in
336           message(ckr.check(tag_inside)))
337
338
339def test_refpage_missing(ckr):
340    """Test the REFPAGE_MISSING message."""
341    ckr.enabled([MessageId.REFPAGE_MISSING])
342    # Should not error: have the tag, an open, and the include
343    assert(not ckr.check(
344        """[open,refpage='%s']
345        --
346        include::{generated}/api/protos/%s.adoc[]""" % (PROTO, PROTO)).messages)
347    assert(not ckr.check(
348        """[open,refpage='%s']
349        --
350        include::{generated}/validity/protos/%s.adoc[]""" % (PROTO, PROTO)).messages)
351
352    # Should not error: manual anchors shouldn't trigger this.
353    assert(not ckr.check("[[%s]]" % PROTO).messages)
354
355    # Should have 1 error: file ends immediately after include
356    assert(ckr.check(
357        "include::{generated}/api/protos/%s.adoc[]" % PROTO).numDiagnostics() == 1)
358    assert(ckr.check(
359        "include::{generated}/validity/protos/%s.adoc[]" % PROTO).numDiagnostics() == 1)
360
361    # Should have 1 error: include is before the refpage open
362    assert(ckr.check(
363        """include::{generated}/api/protos/%s.adoc[]
364        [open,refpage='%s']
365        --""" % (PROTO, PROTO)).numDiagnostics() == 1)
366    assert(ckr.check(
367        """include::{generated}/validity/protos/%s.adoc[]
368        [open,refpage='%s']
369        --""" % (PROTO, PROTO)).numDiagnostics() == 1)
370
371
372def test_refpage_mismatch(ckr):
373    """Test the REFPAGE_MISMATCH message."""
374    ckr.enabled([MessageId.REFPAGE_MISMATCH])
375    # Should not error: have the tag, an open, and a matching include
376    assert(not ckr.check(
377        """[open,refpage='%s']
378        --
379        include::{generated}/api/protos/%s.adoc[]""" % (PROTO, PROTO)).messages)
380    assert(not ckr.check(
381        """[open,refpage='%s']
382        --
383        include::{generated}/validity/protos/%s.adoc[]""" % (PROTO, PROTO)).messages)
384
385    # Should error: have the tag, an open, and a mis-matching include
386    assert(ckr.check(
387        """[open,refpage='%s']
388        --
389        include::{generated}/api/structs/%s.adoc[]""" % (PROTO, STRUCT)).numDiagnostics() == 1)
390    assert(ckr.check(
391        """[open,refpage='%s']
392        --
393        include::{generated}/validity/structs/%s.adoc[]""" % (PROTO, STRUCT)).numDiagnostics() == 1)
394
395
396def test_refpage_unknown_attrib(ckr):
397    """Check the REFPAGE_UNKNOWN_ATTRIB message."""
398    ckr.enabled([MessageId.REFPAGE_UNKNOWN_ATTRIB])
399    # Should not error: these are known attribute names
400    assert(not ckr.check(
401        "[open,refpage='',desc='',type='',xrefs='']").messages)
402
403    # Should error: xref isn't an attribute name.
404    assert(ckr.check(
405        "[open,xref='']").numDiagnostics() == 1)
406
407
408def test_refpage_self_xref(ckr):
409    """Check the REFPAGE_SELF_XREF message."""
410    ckr.enabled([MessageId.REFPAGE_SELF_XREF])
411    # Should not error: not self-referencing
412    assert(not ckr.check(
413        "[open,refpage='abc',xrefs='']").messages)
414    assert(not ckr.check(
415        "[open,refpage='abc',xrefs='123']").messages)
416
417    # Should error: self-referencing isn't an attribute name.
418    assert(loneMsgReplacement(
419        ckr.check("[open,refpage='abc',xrefs='abc']")) == '')
420    assert(loneMsgReplacement(
421        ckr.check("[open,refpage='abc',xrefs='abc 123']")) == '123')
422    assert(loneMsgReplacement(
423        ckr.check("[open,refpage='abc',xrefs='123 abc']")) == '123')
424
425
426def test_refpage_xref_dupe(ckr):
427    """Check the REFPAGE_XREF_DUPE message."""
428    ckr.enabled([MessageId.REFPAGE_XREF_DUPE])
429    # Should not error: no dupes
430    assert(not ckr.check("[open,xrefs='']").messages)
431    assert(not ckr.check("[open,xrefs='123']").messages)
432    assert(not ckr.check("[open,xrefs='abc 123']").messages)
433
434    # Should error: one dupe.
435    assert(loneMsgReplacement(
436        ckr.check("[open,xrefs='abc abc']")) == 'abc')
437    assert(loneMsgReplacement(
438        ckr.check("[open,xrefs='abc   abc']")) == 'abc')
439    assert(loneMsgReplacement(
440        ckr.check("[open,xrefs='abc abc abc']")) == 'abc')
441    assert(loneMsgReplacement(
442        ckr.check("[open,xrefs='abc 123 abc']")) == 'abc 123')
443    assert(loneMsgReplacement(
444        ckr.check("[open,xrefs='123 abc abc']")) == '123 abc')
445
446
447def test_REFPAGE_WHITESPACE(ckr):
448    """Check the REFPAGE_WHITESPACE message."""
449    ckr.enabled([MessageId.REFPAGE_WHITESPACE])
450    # Should not error: no extra whitespace
451    assert(not ckr.check("[open,xrefs='']").messages)
452    assert(not ckr.check("[open,xrefs='123']").messages)
453    assert(not ckr.check("[open,xrefs='abc 123']").messages)
454
455    # Should error: some extraneous whitespace.
456    assert(loneMsgReplacement(
457        ckr.check("[open,xrefs='   \t   ']")) == '')
458    assert(loneMsgReplacement(
459        ckr.check("[open,xrefs='  abc   123  ']")) == 'abc 123')
460    assert(loneMsgReplacement(
461        ckr.check("[open,xrefs='  abc\t123    xyz  ']")) == 'abc 123 xyz')
462
463    # Should *NOT* remove self-reference, just extra whitespace
464    assert(loneMsgReplacement(
465        ckr.check("[open,refpage='abc',xrefs='  abc   123  ']")) == 'abc 123')
466
467    # Even if we turn on the self-reference warning
468    ckr.enabled([MessageId.REFPAGE_WHITESPACE, MessageId.REFPAGE_SELF_XREF])
469    assert(msgReplacement(
470        ckr.check("[open,refpage='abc',xrefs='  abc   123  ']"), 1) == 'abc 123')
471
472
473def test_REFPAGE_DUPLICATE(ckr):
474    """Check the REFPAGE_DUPLICATE message."""
475    ckr.enabled([MessageId.REFPAGE_DUPLICATE])
476    # Should not error: no duplicate refpages.
477    assert(not ckr.check("[open,refpage='abc']").messages)
478    assert(not ckr.check("[open,refpage='123']").messages)
479
480    # Should error: repeated refpage
481    assert(ckr.check(
482        """[open,refpage='abc']
483        [open,refpage='abc']""").messages)
484
485    # Should error: repeated refpage with something intervening
486    assert(ckr.check(
487        """[open,refpage='abc']
488        [open,refpage='123']
489        [open,refpage='abc']""").messages)
490
491
492def test_UNCLOSED_BLOCK(ckr):
493    """Check the UNCLOSED_BLOCK message."""
494    ckr.enabled([MessageId.UNCLOSED_BLOCK])
495    # These should all have 0 errors
496    assert(not ckr.check("== Heading").messages)
497    assert(not ckr.check(
498        """****
499        == Heading
500        ****""").messages)
501    assert(not ckr.check(
502        """****
503        contents
504        ****""").messages)
505    assert(not ckr.check(
506        """****
507        [source,c]
508        ----
509        this is code
510        ----
511        ****""").messages)
512    assert(not ckr.check(
513        """[open,]
514        --
515        123
516
517        [source,c]
518        ----
519        this is code
520        ----
521        ****
522        * this is in a box
523        ****
524
525        Now we can close the ref page.
526        --""").messages)
527
528    # These should all have 1 error because I removed a block close.
529    # Because some of them, the missing block close is an interior one, the stack might look weird,
530    # but it's still only 1 error - no matter how many are left unclosed.
531    assert(ckr.check(
532        """****
533        == Heading""").numDiagnostics() == 1)
534    assert(ckr.check(
535        """****
536        contents""").numDiagnostics() == 1)
537    assert(ckr.check(
538        """****
539        [source,c]
540        ----
541        this is code
542        ****""").numDiagnostics() == 1)
543    assert(ckr.check(
544        """****
545        [source,c]
546        ----
547        this is code
548        ----""").numDiagnostics() == 1)
549    assert(ckr.check(
550        """[open,]
551        --
552        123
553
554        [source,c]
555        ----
556        this is code
557        ----
558        ****
559        * this is in a box
560        ****""").numDiagnostics() == 1)
561    assert(ckr.check(
562        """[open,]
563        --
564        123
565
566        [source,c]
567        ----
568        this is code
569        ----
570        ****
571        * this is in a box
572        --""").numDiagnostics() == 1)
573    assert(ckr.check(
574        """[open,]
575        --
576        123
577
578        [source,c]
579        ----
580        this is code
581        ****
582        * this is in a box
583        ****
584
585        Now we can close the ref page.
586        --""").numDiagnostics() == 1)
587    assert(ckr.check(
588        """[open,]
589        --
590        123
591
592        [source,c]
593        ----
594        this is code
595        ****
596        * this is in a box
597
598        Now we can close the ref page.
599        --""").numDiagnostics() == 1)
600    assert(ckr.check(
601        """[open,]
602        --
603        123
604
605        [source,c]
606        ----
607        this is code
608        ****
609        * this is in a box""").numDiagnostics() == 1)
610
611    # This should have 0 errors of UNCLOSED_BLOCK: the missing opening -- should get automatically fake-inserted,
612    assert(not ckr.check(
613        """[open,]
614        == Heading
615        --""").messages)
616
617    # Should have 1 error: block left open at end of file
618    assert(ckr.check(
619        """[open,]
620        --
621        bla""").numDiagnostics() == 1)
622
623
624def test_code_block_tracking(ckr):
625    """Check to make sure that no other messages get triggered in a code block."""
626    ckr.enabled([MessageId.BAD_ENTITY])
627
628    # Should have 1 error: not a valid entity
629    assert(ckr.check("slink:BogusStruct").numDiagnostics() == 1)
630    assert(ckr.check(
631        """****
632        * slink:BogusStruct
633        ****""").numDiagnostics() == 1)
634
635    # should have zero errors: the invalid entity is inside a code block,
636    # so it shouldn't be parsed.
637    # (In reality, it's mostly the MISSING_MACRO message that might interact with code block tracking,
638    # but this is easier to test in an API-agnostic way.)
639    assert(not ckr.check(
640        """[source,c]
641        ----
642        This code happens to include the characters slink:BogusStruct
643        ----""").messages)
644