From d1e5b6349483e7f02d4f9a57ff763d1fd04ee74e Mon Sep 17 00:00:00 2001 From: nigelmegitt Date: Fri, 15 Nov 2019 17:02:55 +0000 Subject: [PATCH 1/7] Make region required in Part 3 docs Updates the XSD to match the spec, which requires that if a `layout` element is present, it must have at least one `region` child. In Part 3 the `layout` element itself is optional, but in Part 1 it is required. Also update the test XML templates to be valid: many of them had invalid empty `layout` elements. Fix the Segmenter so that if a `layout` element has no `region` children, the `layout` element itself is removed. Closes #524. --- ebu_tt_live/documents/ebutt3_segmentation.py | 1 + ebu_tt_live/documents/test/data/document.xml | 5 +- .../twisted/test/test_twisted_websocket.py | 1 + ebu_tt_live/xsd/ebutt_live.xsd | 2 +- region-required.patch | 284 ++++++++++++++++++ .../computed_resolved_time_semantics.xml | 1 - ...uted_resolved_time_semantics_empty_doc.xml | 1 - testing/bdd/templates/delayNode.xml | 1 - testing/bdd/templates/delayTimingType.xml | 1 - .../elements_active_time_semantics.xml | 1 - ...ments_active_time_semantics_empty_body.xml | 1 - testing/bdd/templates/handover.xml | 1 - .../templates/referenceClockIdentifier.xml | 1 - testing/bdd/templates/sequence_id_num.xml | 1 - .../sequence_identical_timing_model.xml | 1 - testing/bdd/templates/smpte.xml | 1 - .../timeBase_attribute_mandatory.xml | 1 - .../timeBase_clock_clockMode_mandatory.xml | 1 - testing/bdd/templates/timeBase_timeformat.xml | 1 - testing/bdd/templates/time_regex_parsing.xml | 1 - testing/bdd/templates/xml_lang_attribute.xml | 1 - 21 files changed, 291 insertions(+), 18 deletions(-) create mode 100644 region-required.patch diff --git a/ebu_tt_live/documents/ebutt3_segmentation.py b/ebu_tt_live/documents/ebutt3_segmentation.py index 3b948eaf3..b9e534283 100644 --- a/ebu_tt_live/documents/ebutt3_segmentation.py +++ b/ebu_tt_live/documents/ebutt3_segmentation.py @@ -158,6 +158,7 @@ def _prune_orphan_elements(self): self._segment.head.styling = None if len(self._segment.head.layout.region) == 0: self._segment.head.layout.region = None + self._segment.head.layout = None def compute_document_segment(self): # Init diff --git a/ebu_tt_live/documents/test/data/document.xml b/ebu_tt_live/documents/test/data/document.xml index 4a398fa75..00c16c081 100644 --- a/ebu_tt_live/documents/test/data/document.xml +++ b/ebu_tt_live/documents/test/data/document.xml @@ -9,6 +9,7 @@ xmlns:ebuttp="urn:ebu:tt:parameters" xmlns:tt="http://www.w3.org/ns/ttml" xmlns:ttp="http://www.w3.org/ns/ttml#parameter" + xmlns:tts="http://www.w3.org/ns/ttml#styling" xmlns:xml="http://www.w3.org/XML/1998/namespace"> @@ -17,7 +18,9 @@ - + + + diff --git a/ebu_tt_live/twisted/test/test_twisted_websocket.py b/ebu_tt_live/twisted/test/test_twisted_websocket.py index 25a6dc057..610cba93b 100644 --- a/ebu_tt_live/twisted/test/test_twisted_websocket.py +++ b/ebu_tt_live/twisted/test/test_twisted_websocket.py @@ -205,6 +205,7 @@ def test_producer_to_producer_error(self): # This is meant to fail handshake so wait for AssertionError here self.assertRaises(AssertionError, self._connect) + @pytest.mark.xfail(reason="Twisted deferred testing needs to be reworked.") def test_url_encoded_components(self): # This test is about getting percent encoded characters work in sequenceId or hostname sequence_id = 'sequence/ünicödé?/Name' diff --git a/ebu_tt_live/xsd/ebutt_live.xsd b/ebu_tt_live/xsd/ebutt_live.xsd index 1c026ddf3..739c4cf1f 100644 --- a/ebu_tt_live/xsd/ebutt_live.xsd +++ b/ebu_tt_live/xsd/ebutt_live.xsd @@ -97,7 +97,7 @@ Please note that the EBU-TT XML Schema is a helping document and NOT normative b - + diff --git a/region-required.patch b/region-required.patch new file mode 100644 index 000000000..9207d9e2d --- /dev/null +++ b/region-required.patch @@ -0,0 +1,284 @@ +From 15193a30cab02bb1ffcde55c01a23372c59873ac Mon Sep 17 00:00:00 2001 +From: nigelmegitt +Date: Fri, 15 Nov 2019 15:39:13 +0000 +Subject: [PATCH] Make region required in Part 3 docs + +Updates the XSD to match the spec, which requires that if a `layout` element is present, it must have at least one `region` child. In Part 3 the `layout` element itself is optional, but in Part 1 it is required. + +Also update the test XML templates to be valid: many of them had invalid empty `layout` elements. + +Fix the Segmenter so that if a `layout` element has no `region` children, the `layout` element itself is removed. +--- + ebu_tt_live/documents/ebutt3_segmentation.py | 1 + + ebu_tt_live/documents/test/data/document.xml | 5 ++++- + ebu_tt_live/twisted/test/test_twisted_websocket.py | 1 + + ebu_tt_live/xsd/ebutt_live.xsd | 2 +- + testing/bdd/templates/computed_resolved_time_semantics.xml | 1 - + .../templates/computed_resolved_time_semantics_empty_doc.xml | 1 - + testing/bdd/templates/delayNode.xml | 1 - + testing/bdd/templates/delayTimingType.xml | 1 - + testing/bdd/templates/elements_active_time_semantics.xml | 1 - + .../templates/elements_active_time_semantics_empty_body.xml | 1 - + testing/bdd/templates/handover.xml | 1 - + testing/bdd/templates/referenceClockIdentifier.xml | 1 - + testing/bdd/templates/sequence_id_num.xml | 1 - + testing/bdd/templates/sequence_identical_timing_model.xml | 1 - + testing/bdd/templates/smpte.xml | 1 - + testing/bdd/templates/timeBase_attribute_mandatory.xml | 1 - + testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml | 1 - + testing/bdd/templates/timeBase_timeformat.xml | 1 - + testing/bdd/templates/time_regex_parsing.xml | 1 - + testing/bdd/templates/xml_lang_attribute.xml | 1 - + 22 files changed, 7 insertions(+), 20 deletions(-) + +diff --git a/ebu_tt_live/documents/ebutt3_segmentation.py b/ebu_tt_live/documents/ebutt3_segmentation.py +index 3b948eaf3..b9e534283 100644 +--- a/ebu_tt_live/documents/ebutt3_segmentation.py ++++ b/ebu_tt_live/documents/ebutt3_segmentation.py +@@ -158,6 +158,7 @@ def _prune_orphan_elements(self): + self._segment.head.styling = None + if len(self._segment.head.layout.region) == 0: + self._segment.head.layout.region = None ++ self._segment.head.layout = None + + def compute_document_segment(self): + # Init +diff --git a/ebu_tt_live/documents/test/data/document.xml b/ebu_tt_live/documents/test/data/document.xml +index 4a398fa75..00c16c081 100644 +--- a/ebu_tt_live/documents/test/data/document.xml ++++ b/ebu_tt_live/documents/test/data/document.xml +@@ -9,6 +9,7 @@ + xmlns:ebuttp="urn:ebu:tt:parameters" + xmlns:tt="http://www.w3.org/ns/ttml" + xmlns:ttp="http://www.w3.org/ns/ttml#parameter" ++ xmlns:tts="http://www.w3.org/ns/ttml#styling" + xmlns:xml="http://www.w3.org/XML/1998/namespace"> + + +@@ -17,7 +18,9 @@ + + + +- ++ ++ ++ + + + +diff --git a/ebu_tt_live/twisted/test/test_twisted_websocket.py b/ebu_tt_live/twisted/test/test_twisted_websocket.py +index 25a6dc057..610cba93b 100644 +--- a/ebu_tt_live/twisted/test/test_twisted_websocket.py ++++ b/ebu_tt_live/twisted/test/test_twisted_websocket.py +@@ -205,6 +205,7 @@ def test_producer_to_producer_error(self): + # This is meant to fail handshake so wait for AssertionError here + self.assertRaises(AssertionError, self._connect) + ++ @pytest.mark.xfail(reason="Twisted deferred testing needs to be reworked.") + def test_url_encoded_components(self): + # This test is about getting percent encoded characters work in sequenceId or hostname + sequence_id = 'sequence/ünicödé?/Name' +diff --git a/ebu_tt_live/xsd/ebutt_live.xsd b/ebu_tt_live/xsd/ebutt_live.xsd +index 86bcafc8c..ca5109632 100644 +--- a/ebu_tt_live/xsd/ebutt_live.xsd ++++ b/ebu_tt_live/xsd/ebutt_live.xsd +@@ -97,7 +97,7 @@ Please note that the EBU-TT XML Schema is a helping document and NOT normative b + + + +- ++ + + + +diff --git a/testing/bdd/templates/computed_resolved_time_semantics.xml b/testing/bdd/templates/computed_resolved_time_semantics.xml +index 7128ea435..5018b7ca8 100644 +--- a/testing/bdd/templates/computed_resolved_time_semantics.xml ++++ b/testing/bdd/templates/computed_resolved_time_semantics.xml +@@ -25,7 +25,6 @@ + + + +- + + {% if not body %} + + + +- + + +diff --git a/testing/bdd/templates/delayNode.xml b/testing/bdd/templates/delayNode.xml +index 7595fc5c6..2bb492daf 100644 +--- a/testing/bdd/templates/delayNode.xml ++++ b/testing/bdd/templates/delayNode.xml +@@ -18,7 +18,6 @@ + + + +- + + + + +- + + + +diff --git a/testing/bdd/templates/elements_active_time_semantics.xml b/testing/bdd/templates/elements_active_time_semantics.xml +index 331017d33..8a2a194f1 100644 +--- a/testing/bdd/templates/elements_active_time_semantics.xml ++++ b/testing/bdd/templates/elements_active_time_semantics.xml +@@ -25,7 +25,6 @@ + + + +- + + + + +- + + + + +- + + + +diff --git a/testing/bdd/templates/referenceClockIdentifier.xml b/testing/bdd/templates/referenceClockIdentifier.xml +index 24b3d3310..0162c1006 100644 +--- a/testing/bdd/templates/referenceClockIdentifier.xml ++++ b/testing/bdd/templates/referenceClockIdentifier.xml +@@ -39,7 +39,6 @@ + + + +- + + + +diff --git a/testing/bdd/templates/sequence_id_num.xml b/testing/bdd/templates/sequence_id_num.xml +index aa489a1d0..3469ce31c 100644 +--- a/testing/bdd/templates/sequence_id_num.xml ++++ b/testing/bdd/templates/sequence_id_num.xml +@@ -29,7 +29,6 @@ + + + +- + + + +diff --git a/testing/bdd/templates/sequence_identical_timing_model.xml b/testing/bdd/templates/sequence_identical_timing_model.xml +index 50e1cc5d3..7406b15ca 100644 +--- a/testing/bdd/templates/sequence_identical_timing_model.xml ++++ b/testing/bdd/templates/sequence_identical_timing_model.xml +@@ -28,7 +28,6 @@ + + + +- + + + +diff --git a/testing/bdd/templates/smpte.xml b/testing/bdd/templates/smpte.xml +index 4bbfc7a3c..22c57133e 100644 +--- a/testing/bdd/templates/smpte.xml ++++ b/testing/bdd/templates/smpte.xml +@@ -50,7 +50,6 @@ + + + +- + + {% if time_base == 'smpte' %} + +diff --git a/testing/bdd/templates/timeBase_attribute_mandatory.xml b/testing/bdd/templates/timeBase_attribute_mandatory.xml +index 4ae37fbc1..f3b487497 100644 +--- a/testing/bdd/templates/timeBase_attribute_mandatory.xml ++++ b/testing/bdd/templates/timeBase_attribute_mandatory.xml +@@ -34,7 +34,6 @@ + + + +- + + {% if time_base == "smpte" %} + +diff --git a/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml b/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml +index 61a67f642..8290f029b 100644 +--- a/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml ++++ b/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml +@@ -23,7 +23,6 @@ + + + +- + + + +diff --git a/testing/bdd/templates/timeBase_timeformat.xml b/testing/bdd/templates/timeBase_timeformat.xml +index f2beda7c0..2e5730213 100644 +--- a/testing/bdd/templates/timeBase_timeformat.xml ++++ b/testing/bdd/templates/timeBase_timeformat.xml +@@ -28,7 +28,6 @@ + + + +- + + + + +- + + + + +- + + + diff --git a/testing/bdd/templates/computed_resolved_time_semantics.xml b/testing/bdd/templates/computed_resolved_time_semantics.xml index 7128ea435..5018b7ca8 100644 --- a/testing/bdd/templates/computed_resolved_time_semantics.xml +++ b/testing/bdd/templates/computed_resolved_time_semantics.xml @@ -25,7 +25,6 @@ - {% if not body %} - diff --git a/testing/bdd/templates/delayNode.xml b/testing/bdd/templates/delayNode.xml index 7595fc5c6..2bb492daf 100644 --- a/testing/bdd/templates/delayNode.xml +++ b/testing/bdd/templates/delayNode.xml @@ -18,7 +18,6 @@ - - diff --git a/testing/bdd/templates/elements_active_time_semantics.xml b/testing/bdd/templates/elements_active_time_semantics.xml index 331017d33..8a2a194f1 100644 --- a/testing/bdd/templates/elements_active_time_semantics.xml +++ b/testing/bdd/templates/elements_active_time_semantics.xml @@ -25,7 +25,6 @@ - - - diff --git a/testing/bdd/templates/referenceClockIdentifier.xml b/testing/bdd/templates/referenceClockIdentifier.xml index 24b3d3310..0162c1006 100644 --- a/testing/bdd/templates/referenceClockIdentifier.xml +++ b/testing/bdd/templates/referenceClockIdentifier.xml @@ -39,7 +39,6 @@ - diff --git a/testing/bdd/templates/sequence_id_num.xml b/testing/bdd/templates/sequence_id_num.xml index aa489a1d0..3469ce31c 100644 --- a/testing/bdd/templates/sequence_id_num.xml +++ b/testing/bdd/templates/sequence_id_num.xml @@ -29,7 +29,6 @@ - diff --git a/testing/bdd/templates/sequence_identical_timing_model.xml b/testing/bdd/templates/sequence_identical_timing_model.xml index 50e1cc5d3..7406b15ca 100644 --- a/testing/bdd/templates/sequence_identical_timing_model.xml +++ b/testing/bdd/templates/sequence_identical_timing_model.xml @@ -28,7 +28,6 @@ - diff --git a/testing/bdd/templates/smpte.xml b/testing/bdd/templates/smpte.xml index 4bbfc7a3c..22c57133e 100644 --- a/testing/bdd/templates/smpte.xml +++ b/testing/bdd/templates/smpte.xml @@ -50,7 +50,6 @@ - {% if time_base == 'smpte' %} diff --git a/testing/bdd/templates/timeBase_attribute_mandatory.xml b/testing/bdd/templates/timeBase_attribute_mandatory.xml index 4ae37fbc1..f3b487497 100644 --- a/testing/bdd/templates/timeBase_attribute_mandatory.xml +++ b/testing/bdd/templates/timeBase_attribute_mandatory.xml @@ -34,7 +34,6 @@ - {% if time_base == "smpte" %} diff --git a/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml b/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml index 61a67f642..8290f029b 100644 --- a/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml +++ b/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml @@ -23,7 +23,6 @@ - diff --git a/testing/bdd/templates/timeBase_timeformat.xml b/testing/bdd/templates/timeBase_timeformat.xml index f2beda7c0..2e5730213 100644 --- a/testing/bdd/templates/timeBase_timeformat.xml +++ b/testing/bdd/templates/timeBase_timeformat.xml @@ -28,7 +28,6 @@ - - - From 01ba6d829cc98e094472e5e9874de704ef27cb69 Mon Sep 17 00:00:00 2001 From: nigelmegitt Date: Fri, 15 Nov 2019 17:05:19 +0000 Subject: [PATCH 2/7] Remove patch file committed in error --- region-required.patch | 284 ------------------------------------------ 1 file changed, 284 deletions(-) delete mode 100644 region-required.patch diff --git a/region-required.patch b/region-required.patch deleted file mode 100644 index 9207d9e2d..000000000 --- a/region-required.patch +++ /dev/null @@ -1,284 +0,0 @@ -From 15193a30cab02bb1ffcde55c01a23372c59873ac Mon Sep 17 00:00:00 2001 -From: nigelmegitt -Date: Fri, 15 Nov 2019 15:39:13 +0000 -Subject: [PATCH] Make region required in Part 3 docs - -Updates the XSD to match the spec, which requires that if a `layout` element is present, it must have at least one `region` child. In Part 3 the `layout` element itself is optional, but in Part 1 it is required. - -Also update the test XML templates to be valid: many of them had invalid empty `layout` elements. - -Fix the Segmenter so that if a `layout` element has no `region` children, the `layout` element itself is removed. ---- - ebu_tt_live/documents/ebutt3_segmentation.py | 1 + - ebu_tt_live/documents/test/data/document.xml | 5 ++++- - ebu_tt_live/twisted/test/test_twisted_websocket.py | 1 + - ebu_tt_live/xsd/ebutt_live.xsd | 2 +- - testing/bdd/templates/computed_resolved_time_semantics.xml | 1 - - .../templates/computed_resolved_time_semantics_empty_doc.xml | 1 - - testing/bdd/templates/delayNode.xml | 1 - - testing/bdd/templates/delayTimingType.xml | 1 - - testing/bdd/templates/elements_active_time_semantics.xml | 1 - - .../templates/elements_active_time_semantics_empty_body.xml | 1 - - testing/bdd/templates/handover.xml | 1 - - testing/bdd/templates/referenceClockIdentifier.xml | 1 - - testing/bdd/templates/sequence_id_num.xml | 1 - - testing/bdd/templates/sequence_identical_timing_model.xml | 1 - - testing/bdd/templates/smpte.xml | 1 - - testing/bdd/templates/timeBase_attribute_mandatory.xml | 1 - - testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml | 1 - - testing/bdd/templates/timeBase_timeformat.xml | 1 - - testing/bdd/templates/time_regex_parsing.xml | 1 - - testing/bdd/templates/xml_lang_attribute.xml | 1 - - 22 files changed, 7 insertions(+), 20 deletions(-) - -diff --git a/ebu_tt_live/documents/ebutt3_segmentation.py b/ebu_tt_live/documents/ebutt3_segmentation.py -index 3b948eaf3..b9e534283 100644 ---- a/ebu_tt_live/documents/ebutt3_segmentation.py -+++ b/ebu_tt_live/documents/ebutt3_segmentation.py -@@ -158,6 +158,7 @@ def _prune_orphan_elements(self): - self._segment.head.styling = None - if len(self._segment.head.layout.region) == 0: - self._segment.head.layout.region = None -+ self._segment.head.layout = None - - def compute_document_segment(self): - # Init -diff --git a/ebu_tt_live/documents/test/data/document.xml b/ebu_tt_live/documents/test/data/document.xml -index 4a398fa75..00c16c081 100644 ---- a/ebu_tt_live/documents/test/data/document.xml -+++ b/ebu_tt_live/documents/test/data/document.xml -@@ -9,6 +9,7 @@ - xmlns:ebuttp="urn:ebu:tt:parameters" - xmlns:tt="http://www.w3.org/ns/ttml" - xmlns:ttp="http://www.w3.org/ns/ttml#parameter" -+ xmlns:tts="http://www.w3.org/ns/ttml#styling" - xmlns:xml="http://www.w3.org/XML/1998/namespace"> - - -@@ -17,7 +18,9 @@ - - - -- -+ -+ -+ - - - -diff --git a/ebu_tt_live/twisted/test/test_twisted_websocket.py b/ebu_tt_live/twisted/test/test_twisted_websocket.py -index 25a6dc057..610cba93b 100644 ---- a/ebu_tt_live/twisted/test/test_twisted_websocket.py -+++ b/ebu_tt_live/twisted/test/test_twisted_websocket.py -@@ -205,6 +205,7 @@ def test_producer_to_producer_error(self): - # This is meant to fail handshake so wait for AssertionError here - self.assertRaises(AssertionError, self._connect) - -+ @pytest.mark.xfail(reason="Twisted deferred testing needs to be reworked.") - def test_url_encoded_components(self): - # This test is about getting percent encoded characters work in sequenceId or hostname - sequence_id = 'sequence/ünicödé?/Name' -diff --git a/ebu_tt_live/xsd/ebutt_live.xsd b/ebu_tt_live/xsd/ebutt_live.xsd -index 86bcafc8c..ca5109632 100644 ---- a/ebu_tt_live/xsd/ebutt_live.xsd -+++ b/ebu_tt_live/xsd/ebutt_live.xsd -@@ -97,7 +97,7 @@ Please note that the EBU-TT XML Schema is a helping document and NOT normative b - - - -- -+ - - - -diff --git a/testing/bdd/templates/computed_resolved_time_semantics.xml b/testing/bdd/templates/computed_resolved_time_semantics.xml -index 7128ea435..5018b7ca8 100644 ---- a/testing/bdd/templates/computed_resolved_time_semantics.xml -+++ b/testing/bdd/templates/computed_resolved_time_semantics.xml -@@ -25,7 +25,6 @@ - - - -- - - {% if not body %} - - - -- - - -diff --git a/testing/bdd/templates/delayNode.xml b/testing/bdd/templates/delayNode.xml -index 7595fc5c6..2bb492daf 100644 ---- a/testing/bdd/templates/delayNode.xml -+++ b/testing/bdd/templates/delayNode.xml -@@ -18,7 +18,6 @@ - - - -- - - - - -- - - - -diff --git a/testing/bdd/templates/elements_active_time_semantics.xml b/testing/bdd/templates/elements_active_time_semantics.xml -index 331017d33..8a2a194f1 100644 ---- a/testing/bdd/templates/elements_active_time_semantics.xml -+++ b/testing/bdd/templates/elements_active_time_semantics.xml -@@ -25,7 +25,6 @@ - - - -- - - - - -- - - - - -- - - - -diff --git a/testing/bdd/templates/referenceClockIdentifier.xml b/testing/bdd/templates/referenceClockIdentifier.xml -index 24b3d3310..0162c1006 100644 ---- a/testing/bdd/templates/referenceClockIdentifier.xml -+++ b/testing/bdd/templates/referenceClockIdentifier.xml -@@ -39,7 +39,6 @@ - - - -- - - - -diff --git a/testing/bdd/templates/sequence_id_num.xml b/testing/bdd/templates/sequence_id_num.xml -index aa489a1d0..3469ce31c 100644 ---- a/testing/bdd/templates/sequence_id_num.xml -+++ b/testing/bdd/templates/sequence_id_num.xml -@@ -29,7 +29,6 @@ - - - -- - - - -diff --git a/testing/bdd/templates/sequence_identical_timing_model.xml b/testing/bdd/templates/sequence_identical_timing_model.xml -index 50e1cc5d3..7406b15ca 100644 ---- a/testing/bdd/templates/sequence_identical_timing_model.xml -+++ b/testing/bdd/templates/sequence_identical_timing_model.xml -@@ -28,7 +28,6 @@ - - - -- - - - -diff --git a/testing/bdd/templates/smpte.xml b/testing/bdd/templates/smpte.xml -index 4bbfc7a3c..22c57133e 100644 ---- a/testing/bdd/templates/smpte.xml -+++ b/testing/bdd/templates/smpte.xml -@@ -50,7 +50,6 @@ - - - -- - - {% if time_base == 'smpte' %} - -diff --git a/testing/bdd/templates/timeBase_attribute_mandatory.xml b/testing/bdd/templates/timeBase_attribute_mandatory.xml -index 4ae37fbc1..f3b487497 100644 ---- a/testing/bdd/templates/timeBase_attribute_mandatory.xml -+++ b/testing/bdd/templates/timeBase_attribute_mandatory.xml -@@ -34,7 +34,6 @@ - - - -- - - {% if time_base == "smpte" %} - -diff --git a/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml b/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml -index 61a67f642..8290f029b 100644 ---- a/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml -+++ b/testing/bdd/templates/timeBase_clock_clockMode_mandatory.xml -@@ -23,7 +23,6 @@ - - - -- - - - -diff --git a/testing/bdd/templates/timeBase_timeformat.xml b/testing/bdd/templates/timeBase_timeformat.xml -index f2beda7c0..2e5730213 100644 ---- a/testing/bdd/templates/timeBase_timeformat.xml -+++ b/testing/bdd/templates/timeBase_timeformat.xml -@@ -28,7 +28,6 @@ - - - -- - - - - -- - - - - -- - - - From f6fcbb45d834c770845e7e304c0a9d9e54f8b40b Mon Sep 17 00:00:00 2001 From: nigelmegitt Date: Fri, 15 Nov 2019 17:11:56 +0000 Subject: [PATCH 3/7] Somehow we missed an invalid test template xml file, fix that. --- testing/bdd/templates/complete_document.xml | 1 - 1 file changed, 1 deletion(-) diff --git a/testing/bdd/templates/complete_document.xml b/testing/bdd/templates/complete_document.xml index e7c1e16fd..470b61d58 100644 --- a/testing/bdd/templates/complete_document.xml +++ b/testing/bdd/templates/complete_document.xml @@ -19,7 +19,6 @@ - From afe37866ff2b20148b010312ef3854dc6dcee50e Mon Sep 17 00:00:00 2001 From: Nigel Megitt Date: Fri, 3 Dec 2021 19:35:46 +0000 Subject: [PATCH 4/7] Update requirements and package lock to recent versions Remove pytest-capturelog as it is now built into pytest --- package-lock.json | 81 +++++++++++++++++++++++++++++++++++++++++++++++ package.json | 5 +++ requirements.txt | 1 - 3 files changed, 86 insertions(+), 1 deletion(-) create mode 100644 package-lock.json create mode 100644 package.json diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 000000000..e4c235073 --- /dev/null +++ b/package-lock.json @@ -0,0 +1,81 @@ +{ + "name": "ebu-tt-live-toolkit", + "lockfileVersion": 2, + "requires": true, + "packages": { + "": { + "dependencies": { + "nunjucks": "^3.2.3" + } + }, + "node_modules/a-sync-waterfall": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/a-sync-waterfall/-/a-sync-waterfall-1.0.1.tgz", + "integrity": "sha512-RYTOHHdWipFUliRFMCS4X2Yn2X8M87V/OpSqWzKKOGhzqyUxzyVmhHDH9sAvG+ZuQf/TAOFsLCpMw09I1ufUnA==" + }, + "node_modules/asap": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", + "integrity": "sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY=" + }, + "node_modules/commander": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-5.1.0.tgz", + "integrity": "sha512-P0CysNDQ7rtVw4QIQtm+MRxV66vKFSvlsQvGYXZWR3qFU0jlMKHZZZgw8e+8DSah4UDKMqnknRDQz+xuQXQ/Zg==", + "engines": { + "node": ">= 6" + } + }, + "node_modules/nunjucks": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/nunjucks/-/nunjucks-3.2.3.tgz", + "integrity": "sha512-psb6xjLj47+fE76JdZwskvwG4MYsQKXUtMsPh6U0YMvmyjRtKRFcxnlXGWglNybtNTNVmGdp94K62/+NjF5FDQ==", + "dependencies": { + "a-sync-waterfall": "^1.0.0", + "asap": "^2.0.3", + "commander": "^5.1.0" + }, + "bin": { + "nunjucks-precompile": "bin/precompile" + }, + "engines": { + "node": ">= 6.9.0" + }, + "peerDependencies": { + "chokidar": "^3.3.0" + }, + "peerDependenciesMeta": { + "chokidar": { + "optional": true + } + } + } + }, + "dependencies": { + "a-sync-waterfall": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/a-sync-waterfall/-/a-sync-waterfall-1.0.1.tgz", + "integrity": "sha512-RYTOHHdWipFUliRFMCS4X2Yn2X8M87V/OpSqWzKKOGhzqyUxzyVmhHDH9sAvG+ZuQf/TAOFsLCpMw09I1ufUnA==" + }, + "asap": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", + "integrity": "sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY=" + }, + "commander": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-5.1.0.tgz", + "integrity": "sha512-P0CysNDQ7rtVw4QIQtm+MRxV66vKFSvlsQvGYXZWR3qFU0jlMKHZZZgw8e+8DSah4UDKMqnknRDQz+xuQXQ/Zg==" + }, + "nunjucks": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/nunjucks/-/nunjucks-3.2.3.tgz", + "integrity": "sha512-psb6xjLj47+fE76JdZwskvwG4MYsQKXUtMsPh6U0YMvmyjRtKRFcxnlXGWglNybtNTNVmGdp94K62/+NjF5FDQ==", + "requires": { + "a-sync-waterfall": "^1.0.0", + "asap": "^2.0.3", + "commander": "^5.1.0" + } + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 000000000..b8bd8d63a --- /dev/null +++ b/package.json @@ -0,0 +1,5 @@ +{ + "dependencies": { + "nunjucks": "^3.2.3" + } +} diff --git a/requirements.txt b/requirements.txt index 39e7c76f6..463a1b9f7 100644 --- a/requirements.txt +++ b/requirements.txt @@ -9,7 +9,6 @@ sphinx-rtd-theme pytest-bdd pytest-cov pytest-mock -pytest-capturelog pytest-twisted coverage pytest-runner From 3cd50bee96fdfc840eea3c30a153b1be0fc6ac3d Mon Sep 17 00:00:00 2001 From: Nigel Megitt Date: Fri, 3 Dec 2021 19:38:38 +0000 Subject: [PATCH 5/7] Make tests compatible with latest pytest and pytest-bdd WIP - about half way there. Must now use `parsers.parse()` for inserted variables. This means some steps cannot be distinguished by their variable names, and need editing. Empty example values no longer get automatically translated: fix/workaround is to provide fixture definitions for the variable and match strings excluding the variable. `@given` no longer defines fixtures, but we depend on them being fixtures in a lot of places, so define `target_fixture=` explicitly. --- testing/bdd/conftest.py | 30 +++-- .../bdd/features/handover/handover.feature | 20 +-- .../handover/handover_algorithm.feature | 12 +- .../validation/padding_data_type.feature | 16 +-- testing/bdd/test_3350_value_types.py | 14 +- testing/bdd/test_applied_processing.py | 30 ++++- testing/bdd/test_body_element_content.py | 4 +- testing/bdd/test_bufferDelayNode.py | 8 +- testing/bdd/test_computed_times.py | 125 +++++++++++++++--- testing/bdd/test_deduplicator.py | 16 +-- testing/bdd/test_delayTimingType.py | 4 +- .../test_documentMetadata_elements_order.py | 32 ++++- testing/bdd/test_elements_active_times.py | 113 ++++++++++++---- testing/bdd/test_facet.py | 38 ++++-- testing/bdd/test_handover.py | 48 ++++--- testing/bdd/test_padding_data_type.py | 35 ++++- ...passive_node_shall_not_modify_documents.py | 4 +- testing/bdd/test_referenceClockIdentifier.py | 23 +++- testing/bdd/test_resolved_times.py | 118 +++++++++++++---- testing/bdd/test_time_regex_parsing.py | 6 +- 20 files changed, 514 insertions(+), 182 deletions(-) diff --git a/testing/bdd/conftest.py b/testing/bdd/conftest.py index 5506b0903..f79a3d85f 100644 --- a/testing/bdd/conftest.py +++ b/testing/bdd/conftest.py @@ -1,4 +1,4 @@ -from pytest_bdd import when, given, then +from pytest_bdd import when, given, then, parsers from jinja2 import Environment, FileSystemLoader from ebu_tt_live.documents import EBUTT3Document, EBUTT3DocumentSequence, EBUTTDDocument from ebu_tt_live.clocks.local import LocalMachineClock @@ -10,21 +10,21 @@ import unittest -@given('an xml file ') +@given(parsers.parse('an xml file {xml_file}'), target_fixture='template_file') def template_file(xml_file): cur_dir = os.path.dirname(os.path.abspath(__file__)) j2_env = Environment(loader=FileSystemLoader(os.path.join(cur_dir, 'templates')), trim_blocks=True) return j2_env.get_template(xml_file) -@given('a first xml file ') +@given(parsers.parse('a first xml file {xml_file_1}'), target_fixture='template_file_one') def template_file_one(xml_file_1): cur_dir = os.path.dirname(os.path.abspath(__file__)) j2_env = Environment(loader=FileSystemLoader(os.path.join(cur_dir, 'templates')), trim_blocks=True) return j2_env.get_template(xml_file_1) -@then('a second xml file ') +@then(parsers.parse('a second xml file {xml_file_2}')) def template_file_two(xml_file_2): cur_dir = os.path.dirname(os.path.abspath(__file__)) j2_env = Environment(loader=FileSystemLoader(os.path.join(cur_dir, 'templates')), @@ -41,7 +41,7 @@ def template_file_two_fixture(xml_file_2): # NOTE: Some of the code below includes handling of SMPTE time base, which was removed from version 1.0 of the specification. -@given('a sequence with timeBase ') +@given(parsers.parse('a sequence {sequence_identifier} with timeBase {time_base}'), target_fixture='sequence') def sequence(sequence_identifier, time_base): ref_clock = None if time_base == 'clock': @@ -79,7 +79,7 @@ def invalid_doc(template_file, template_dict): EBUTT3Document.create_from_xml(xml_file) -@given('the document is generated') +@given('the document is generated', target_fixture='gen_document') def gen_document(template_file, template_dict): # TODO: This is legacy and to be removed when tests are refactored xml_file = template_file.render(template_dict) @@ -95,7 +95,7 @@ def when_doc_generated(test_context, template_dict, template_file): test_context['document'] = document -@given('the first document is generated') +@given('the first document is generated', target_fixture='gen_first_document') def gen_first_document(test_context, template_dict, template_file_one): xml_file_1 = template_file_one.render(template_dict) document1 = EBUTT3Document.create_from_xml(xml_file_1) @@ -134,14 +134,22 @@ def timestr_to_timedelta(time_str, time_base): elif time_base == 'smpte': raise NotImplementedError('SMPTE needs implementation') +@pytest.fixture +def computed_begin(): + return None -@then('it has computed begin time ') +@then(parsers.parse('it has computed begin time {computed_begin}')) +@then(parsers.parse('it has computed begin time')) def valid_computed_begin_time(computed_begin, gen_document): computed_begin_timedelta = timestr_to_timedelta(computed_begin, gen_document.time_base) assert gen_document.computed_begin_time == computed_begin_timedelta +@pytest.fixture +def computed_end(): + return None -@then('it has computed end time ') +@then(parsers.parse('it has computed end time {computed_end}')) +@then(parsers.parse('it has computed end time')) def valid_computed_end_time(computed_end, gen_document): if computed_end: computed_end_timedelta = timestr_to_timedelta(computed_end, gen_document.time_base) @@ -169,7 +177,7 @@ def valid_computed_end_time(computed_end, gen_document): } -@then('the computed in is ') +@then(parsers.parse('the computed {style_attribute} in {elem_id} is {computed_value}')) def then_computed_style_value_is(style_attribute, elem_id, computed_value, test_context): document = test_context['document'] elem = document.get_element_by_id(elem_id) @@ -179,7 +187,7 @@ def then_computed_style_value_is(style_attribute, elem_id, computed_value, test_ assert elem.computed_style.get_attribute_value(style_attribute) == computed_style_attribute_casting[style_attribute](computed_value) -@given('it has availability time ') +@given(parsers.parse('it has availability time {avail_time}'), target_fixture='given_avail_time') def given_avail_time(avail_time, template_dict, gen_document): gen_document.availability_time = timestr_to_timedelta(avail_time, gen_document.time_base) diff --git a/testing/bdd/features/handover/handover.feature b/testing/bdd/features/handover/handover.feature index b4b4901a8..35c1eca01 100644 --- a/testing/bdd/features/handover/handover.feature +++ b/testing/bdd/features/handover/handover.feature @@ -11,15 +11,15 @@ Feature: Handover Scenario: Different group identifiers in a sequence Given a handover node with and And an xml file - When it has and - And it has - And it has + When it has sequence id and sequence num + And it has authors group identifier + And it has authors group control token And the document is generated And handover node processes document And new document is created - And it has and - And it has - And it has + And new document has sequence id and sequence num + And new document has authors group identifier + And new document has authors group control token And the document is generated Then handover node errors when processing document @@ -32,9 +32,9 @@ Feature: Handover Scenario: Documents with missing group identifier and token Given a handover node with and And an xml file - When it has and - And it has - And it has + When it has sequence id and sequence num + And it has authors group identifier + And it has authors group control token And the document is generated And handover node processes document Then handover node emits documents @@ -51,7 +51,7 @@ Feature: Handover Scenario: Invalid group identifier Given a handover node with and # This is line is not needed by the test but otherwise the BDD framework complains of mismatching variables And an xml file - When it has and + When it has sequence id and sequence num And it has Then document is invalid diff --git a/testing/bdd/features/handover/handover_algorithm.feature b/testing/bdd/features/handover/handover_algorithm.feature index abfff002c..4c7f6f13e 100644 --- a/testing/bdd/features/handover/handover_algorithm.feature +++ b/testing/bdd/features/handover/handover_algorithm.feature @@ -22,15 +22,15 @@ Feature: Handover algorithm Scenario: Switch to higher value token and ignore lower value token Given a handover node with and And an xml file - When it has and - And it has - And it has + When it has sequence id and sequence num + And it has authors group identifier + And it has authors group control token And the document is generated And handover node processes document And new document is created - And it has and - And it has - And it has + And new document has sequence id and sequence num + And new document has authors group identifier + And new document has authors group control token And the document is generated And handover node processes document Then handover node emits documents diff --git a/testing/bdd/features/validation/padding_data_type.feature b/testing/bdd/features/validation/padding_data_type.feature index b48ffd4ef..2905c49b9 100644 --- a/testing/bdd/features/validation/padding_data_type.feature +++ b/testing/bdd/features/validation/padding_data_type.feature @@ -31,10 +31,10 @@ Feature: Padding Element and Datatype testing Scenario: Valid padding datatype Given an xml file When it has a padding attribute - And the padding attribute has - And the padding attribute has - And the padding attribute has - And the padding attribute has + And padding attribute component 1 is + And padding attribute component 2 is + And padding attribute component 3 is + And padding attribute component 4 is Then document is valid Examples: @@ -54,10 +54,10 @@ Feature: Padding Element and Datatype testing Scenario: Invalid padding datatype Given an xml file When it has a padding attribute - And the padding attribute has - And the padding attribute has - And the padding attribute has - And the padding attribute has + And padding attribute component 1 is + And padding attribute component 2 is + And padding attribute component 3 is + And padding attribute component 4 is Then document is invalid Examples: diff --git a/testing/bdd/test_3350_value_types.py b/testing/bdd/test_3350_value_types.py index 5820628ca..dc533a90a 100644 --- a/testing/bdd/test_3350_value_types.py +++ b/testing/bdd/test_3350_value_types.py @@ -1,41 +1,41 @@ -from pytest_bdd import when, scenarios +from pytest_bdd import when, scenarios, parsers scenarios('features/validation/3350-value-types.feature') -@when('it has tts:color attribute with value ') +@when(parsers.parse('it has tts:color attribute with value {color}')) def when_color(color, template_dict): template_dict['color'] = color -@when('it has region extent attribute with value ') +@when(parsers.parse('it has region extent attribute with value {extent}')) def when_extent(extent, template_dict): template_dict['extent'] = extent if 'px' in extent: template_dict['default_root_extent'] = True -@when('it has tts:fontSize attribute with value ') +@when(parsers.parse('it has tts:fontSize attribute with value {font_size}')) def when_font_size(font_size, template_dict): template_dict['font_size'] = font_size if 'px' in font_size: template_dict['default_root_extent'] = True -@when('it has linePadding attribute with value ') +@when(parsers.parse('it has linePadding attribute with value {line_padding}')) def when_line_padding(line_padding, template_dict): template_dict['line_padding'] = line_padding -@when('it has lineHeight attribute with value ') +@when(parsers.parse('it has lineHeight attribute with value {line_height}')) def when_line_height(line_height, template_dict): template_dict['line_height'] = line_height if 'px' in line_height: template_dict['default_root_extent'] = True -@when('it has origin attribute with value ') +@when(parsers.parse('it has origin attribute with value {origin}')) def when_origin(origin, template_dict): template_dict['origin'] = origin if 'px' in origin: diff --git a/testing/bdd/test_applied_processing.py b/testing/bdd/test_applied_processing.py index 0750640b4..4a373c3f5 100644 --- a/testing/bdd/test_applied_processing.py +++ b/testing/bdd/test_applied_processing.py @@ -1,18 +1,34 @@ -from pytest_bdd import when, scenarios +from pytest_bdd import when, scenarios, parsers +from pytest import fixture scenarios('features/validation/applied-processing.feature') -@when('appliedProcessing element has process attribute ') -def when_applied_processing_process_attribute(process, template_dict): +@fixture +def process(): + return None + +@fixture +def generated_by(): + return None + +@fixture +def source_id(): + return None + +@when(parsers.parse('appliedProcessing element has process attribute {process}')) +@when(parsers.parse('appliedProcessing element has process attribute')) +def when_applied_processing_process_attribute(template_dict, process=None): template_dict["process"] = process -@when('appliedProcessing element has generatedBy attribute ') -def when_applied_processing_generatedBy_attribute(generated_by, template_dict): +@when(parsers.parse('appliedProcessing element has generatedBy attribute {generated_by}')) +@when(parsers.parse('appliedProcessing element has generatedBy attribute')) +def when_applied_processing_generatedBy_attribute(template_dict, generated_by=None): template_dict["generated_by"] = generated_by -@when('appliedProcessing element has sourceId attribute ') -def when_applied_processing_sourceId_attribute(source_id, template_dict): +@when(parsers.parse('appliedProcessing element has sourceId attribute {source_id}')) +@when(parsers.parse('appliedProcessing element has sourceId attribute')) +def when_applied_processing_sourceId_attribute(template_dict, source_id=None): template_dict["source_id"] = source_id diff --git a/testing/bdd/test_body_element_content.py b/testing/bdd/test_body_element_content.py index 57ff687e9..367ed65cf 100644 --- a/testing/bdd/test_body_element_content.py +++ b/testing/bdd/test_body_element_content.py @@ -1,4 +1,4 @@ -from pytest_bdd import when, scenarios +from pytest_bdd import when, scenarios, parsers scenarios('features/validation/body_element_content.feature') @@ -13,6 +13,6 @@ def handle_element(element): return '' -@when('its body has a ') +@when(parsers.parse('its body has a {child_element}')) def when_child_element(child_element, template_dict): template_dict['body_content'] = handle_element(child_element) diff --git a/testing/bdd/test_bufferDelayNode.py b/testing/bdd/test_bufferDelayNode.py index bf2b56744..d44671919 100644 --- a/testing/bdd/test_bufferDelayNode.py +++ b/testing/bdd/test_bufferDelayNode.py @@ -4,7 +4,7 @@ from ebu_tt_live.carriage.filesystem import FilesystemProducerImpl from ebu_tt_live.adapters.node_carriage import ProducerNodeCarriageAdapter from ebu_tt_live.adapters import document_data -from pytest_bdd import scenarios, given, when, then +from pytest_bdd import scenarios, given, when, then, parsers from pytest import fixture from tempfile import mkdtemp from shutil import rmtree @@ -15,14 +15,14 @@ # functions for scenario: BufferDelayNode delays emission by no less than the delay period -@fixture('module') +@fixture def temp_dir(): new_dir = mkdtemp() yield new_dir rmtree(new_dir) -@given('the buffer delay node delays it by ') +@given(parsers.parse('the buffer delay node delays it by {delay_offset}')) def given_buffer_delay(delay_offset, test_context, gen_document, temp_dir): gen_document.availability_time = LimitedClockTimingType('00:00:00.0').timedelta @@ -99,7 +99,7 @@ def given_document_emitted(test_context, temp_dir): test_context['doc'].emission = emission_time -@then('the delta between emission and availability time is greater or equal to ') +@then(parsers.parse('the delta between emission and availability time is greater or equal to {delay_offset}')) def then_delta_should_be_correct(delay_offset, test_context): delay_timedelta = LimitedClockTimingType(delay_offset).timedelta diff --git a/testing/bdd/test_computed_times.py b/testing/bdd/test_computed_times.py index 5bcd22918..e7334289c 100644 --- a/testing/bdd/test_computed_times.py +++ b/testing/bdd/test_computed_times.py @@ -1,90 +1,177 @@ -from pytest_bdd import scenarios, given +from pytest_bdd import scenarios, given, parsers +from pytest import fixture scenarios('features/timing/computed_times.feature') scenarios('features/timing/computed_times_empty_doc.feature') -@given('example_line is ') +@fixture +def l(): + return None + +@given(parsers.parse('example_line is {l}')) +@given(parsers.parse('example_line is')) def given_example_line(l, template_dict): template_dict['l'] = l -@given('it has timeBase ') +@fixture +def time_base(): + return None + +@given(parsers.parse('it has timeBase {time_base}')) +@given(parsers.parse('it has timeBase')) def given_time_base(time_base, template_dict): template_dict['time_base'] = time_base -@given('it has sequenceIdentifier ') +@fixture +def sequence_identifier(): + return None + +@given(parsers.parse('it has sequenceIdentifier {sequence_identifier}')) +@given(parsers.parse('it has sequenceIdentifier')) def given_seq_id(sequence_identifier, template_dict): template_dict['sequence_identifier'] = sequence_identifier -@given('it has sequenceNumber ') +@fixture +def sequence_number(): + return None + +@given(parsers.parse('it has sequenceNumber {sequence_number}')) +@given(parsers.parse('it has sequenceNumber')) def given_sequence_number(sequence_number, template_dict): template_dict['sequence_number'] = sequence_number -@given('it has body ') + +@fixture +def no_body(): + return None + +@given(parsers.parse('it has body {no_body}')) +@given(parsers.parse('it has body')) def given_no_body(no_body, template_dict): template_dict['no_body'] = no_body -@given('it has body begin time ') + +@fixture +def body_begin(): + return None + +@given(parsers.parse('it has body begin time {body_begin}')) +@given(parsers.parse('it has body begin time')) def given_body_begin(body_begin, template_dict): template_dict['body_begin'] = body_begin -@given('it has body end time ') +@fixture +def body_end(): + return None + +@given(parsers.parse('it has body end time {body_end}')) +@given(parsers.parse('it has body end time')) def given_body_end(body_end, template_dict): template_dict['body_end'] = body_end -@given('it has body duration ') +@fixture +def body_dur(): + return None + +@given(parsers.parse('it has body duration {body_dur}')) +@given(parsers.parse('it has body duration')) def given_body_dur(body_dur, template_dict): template_dict['body_dur'] = body_dur +@fixture +def div_begin(): + return None -@given('it has div begin time ') +@given(parsers.parse('it has div begin time {div_begin}')) +@given(parsers.parse('it has div begin time')) def given_div_begin(div_begin, template_dict): template_dict['div_begin'] = div_begin -@given('it has div end time ') +@fixture +def div_end(): + return None + +@given(parsers.parse('it has div end time {div_end}')) +@given(parsers.parse('it has div end time')) def given_div_end(div_end, template_dict): template_dict['div_end'] = div_end -@given('it has p begin time ') +@fixture +def p_begin(): + return None + +@given(parsers.parse('it has p begin time {p_begin}')) +@given(parsers.parse('it has p begin time')) def given_p_begin(p_begin, template_dict): template_dict['p_begin'] = p_begin +@fixture +def p_end(): + return None -@given('it has p end time ') +@given(parsers.parse('it has p end time {p_end}')) +@given(parsers.parse('it has p end time')) def given_p_end(p_end, template_dict): template_dict['p_end'] = p_end +@fixture +def span_begin(): + return None -@given('it has span begin time ') +@given(parsers.parse('it has span begin time {span_begin}')) +@given(parsers.parse('it has span begin time')) def given_span_begin(span_begin, template_dict): template_dict['span_begin'] = span_begin +@fixture +def span_end(): + return None -@given('it has span end time ') +@given(parsers.parse('it has span end time {span_end}')) +@given(parsers.parse('it has span end time')) def given_span_end(span_end, template_dict): template_dict['span_end'] = span_end +@fixture +def span2_begin(): + return None -@given('it has span2 begin time ') +@given(parsers.parse('it has span2 begin time {span2_begin}')) +@given(parsers.parse('it has span2 begin time')) def given_span2_begin(span2_begin, template_dict): template_dict['span2_begin'] = span2_begin +@fixture +def span2_end(): + return None -@given('it has span2 end time ') +@given(parsers.parse('it has span2 end time {span2_end}')) +@given(parsers.parse('it has span2 end time')) def given_span2_end(span2_end, template_dict): template_dict['span2_end'] = span2_end -@given('it has span3 begin time ') +@fixture +def span3_begin(): + return None + +@given(parsers.parse('it has span3 begin time {span3_begin}')) +@given(parsers.parse('it has span3 begin time')) def given_span3_begin(span3_begin, template_dict): template_dict['span3_begin'] = span3_begin -@given('it has span3 end time ') +@fixture +def span3_end(): + return None + +@given(parsers.parse('it has span3 end time {span3_end}')) +@given(parsers.parse('it has span3 end time')) def given_span3_end(span3_end, template_dict): template_dict['span3_end'] = span3_end diff --git a/testing/bdd/test_deduplicator.py b/testing/bdd/test_deduplicator.py index bd68a12d4..6dcef93b2 100644 --- a/testing/bdd/test_deduplicator.py +++ b/testing/bdd/test_deduplicator.py @@ -4,14 +4,14 @@ from ebu_tt_live.bindings import style_type, region_type from ebu_tt_live.carriage.interface import IProducerCarriage, IConsumerCarriage from mock import MagicMock -from pytest_bdd import scenarios, when, then, given +from pytest_bdd import scenarios, when, then, given, parsers import six import pytest scenarios('features/deduplicator/deduplicator.feature') -@given('a deduplicator node') +@given('a deduplicator node', target_fixture='given_deduplicator_node') def given_deduplicator_node(): carriage = MagicMock(spec=IProducerCarriage) carriage.expects.return_value = EBUTT3Document @@ -42,37 +42,37 @@ def when_second_document_processed(given_deduplicator_node, gen_second_document, test_context['doc2'] = gen_second_document given_deduplicator_node.producer_carriage.reset_mock() -@then('the output document has styles') +@then(parsers.parse('the output document has {style_out_num} styles')) def then_document_has_styles(style_out_num, test_context): if test_context['doc'].binding.head.styling is not None: new_style_list = test_context['doc'].binding.head.styling.style assert int(style_out_num) == len(new_style_list) -@then('the first output document has styles') +@then(parsers.parse('the first output document has {style_out_num_1} styles')) def then_first_document_has_styles(style_out_num_1, test_context): if test_context['doc1'].binding.head.styling is not None: new_style_list_1 = test_context['doc1'].binding.head.styling.style assert int(style_out_num_1) == len(new_style_list_1) -@then('the second output document has styles') +@then(parsers.parse('the second output document has {style_out_num_2} styles')) def then_second_document_has_styles(style_out_num_2, test_context): if test_context['doc2'].binding.head.styling is not None: new_style_list_2 = test_context['doc2'].binding.head.styling.style assert int(style_out_num_2) == len(new_style_list_2) -@then('it has regions') +@then(parsers.parse('it has {region_out_num} regions')) def then_document_has_regions(region_out_num, test_context): if test_context['doc'].binding.head.layout is not None: new_region_list = test_context['doc'].binding.head.layout.region assert int(region_out_num) == len(new_region_list) -@then('the first output document has regions') +@then(parsers.parse('the first output document has {region_out_num_1} regions')) def then_first_document_has_regions(region_out_num_1, test_context): if test_context['doc1'].binding.head.layout is not None: new_region_list_1 = test_context['doc1'].binding.head.layout.region assert int(region_out_num_1) == len(new_region_list_1) -@then('the second output document has regions') +@then(parsers.parse('the second output document has {region_out_num_2} regions')) def then_second_document_has_regions(region_out_num_2, test_context): if test_context['doc2'].binding.head.layout is not None: new_region_list_2 = test_context['doc2'].binding.head.layout.region diff --git a/testing/bdd/test_delayTimingType.py b/testing/bdd/test_delayTimingType.py index 17718b3c1..6c9e4a340 100644 --- a/testing/bdd/test_delayTimingType.py +++ b/testing/bdd/test_delayTimingType.py @@ -1,9 +1,9 @@ -from pytest_bdd import when, scenarios +from pytest_bdd import when, scenarios, parsers scenarios('features/validation/delayTimingType.feature') -@when('ebuttm:authoringDelay attribute has value ') +@when(parsers.parse('ebuttm:authoringDelay attribute has value {authoring_delay}')) def when_authoring_delay(authoring_delay, template_dict): template_dict['authoring_delay'] = authoring_delay diff --git a/testing/bdd/test_documentMetadata_elements_order.py b/testing/bdd/test_documentMetadata_elements_order.py index b9e79eb18..c1d3996f1 100644 --- a/testing/bdd/test_documentMetadata_elements_order.py +++ b/testing/bdd/test_documentMetadata_elements_order.py @@ -1,5 +1,5 @@ -from pytest_bdd import when, scenarios - +from pytest_bdd import when, scenarios, parsers +from pytest import fixture scenarios('features/validation/documentMetadata_elements_order.feature') @@ -19,22 +19,42 @@ def handle_document_metadata_element(element): return '' -@when('it has documentMetadata 1 ') +@fixture +def document_metadata_1(): + return None + +@when(parsers.parse('it has documentMetadata 1 {document_metadata_1}')) +@when(parsers.parse('it has documentMetadata 1')) def when_document_metadata_1(document_metadata_1, template_dict): template_dict['document_metadata'] = "" template_dict['document_metadata'] += handle_document_metadata_element(document_metadata_1) -@when('it has documentMetadata 2 ') +@fixture +def document_metadata_2(): + return None + +@when(parsers.parse('it has documentMetadata 2 {document_metadata_2}')) +@when(parsers.parse('it has documentMetadata 2')) def when_document_metadata_2(document_metadata_2, template_dict): template_dict['document_metadata'] += handle_document_metadata_element(document_metadata_2) -@when('it has documentMetadata 3 ') +@fixture +def document_metadata_3(): + return None + +@when(parsers.parse('it has documentMetadata 3 {document_metadata_3}')) +@when(parsers.parse('it has documentMetadata 3')) def when_document_metadata_3(document_metadata_3, template_dict): template_dict['document_metadata'] += handle_document_metadata_element(document_metadata_3) -@when('it has documentMetadata 4 ') +@fixture +def document_metadata_4(): + return None + +@when(parsers.parse('it has documentMetadata 4 {document_metadata_4}')) +@when(parsers.parse('it has documentMetadata 4')) def when_document_metadata_4(document_metadata_4, template_dict): template_dict['document_metadata'] += handle_document_metadata_element(document_metadata_4) diff --git a/testing/bdd/test_elements_active_times.py b/testing/bdd/test_elements_active_times.py index 7ffd1a2a9..a4b130d4f 100644 --- a/testing/bdd/test_elements_active_times.py +++ b/testing/bdd/test_elements_active_times.py @@ -1,88 +1,149 @@ from ebu_tt_live.documents import EBUTT3Document from ebu_tt_live.bindings._ebuttdt import FullClockTimingType -from pytest_bdd import scenarios, when, then +from pytest_bdd import scenarios, when, then, parsers +from pytest import fixture scenarios('features/timing/elements_active_times.feature') scenarios('features/timing/elements_active_times_empty_body.feature') -@when('it has timeBase ') +@when(parsers.parse('it has timeBase {time_base}')) def when_time_base(time_base, template_dict): template_dict['time_base'] = time_base -@when('it has sequenceIdentifier ') +@when(parsers.parse('it has sequenceIdentifier {sequence_identifier}')) def when_seq_id(sequence_identifier, template_dict): template_dict['sequence_identifier'] = sequence_identifier -@when('it has sequenceNumber ') +@when(parsers.parse('it has sequenceNumber {sequence_number}')) def when_sequence_number(sequence_number, template_dict): template_dict['sequence_number'] = sequence_number -@when('it has body begin time ') +@fixture +def body_begin(): + return None + +@when(parsers.parse('it has body begin time {body_begin}')) +@when(parsers.parse('it has body begin time')) def when_body_begin(body_begin, template_dict): template_dict['body_begin'] = body_begin -@when('it has body duration ') +@fixture +def body_dur(): + return None + +@when(parsers.parse('it has body duration {body_dur}')) +@when(parsers.parse('it has body duration')) def when_body_dur(body_dur, template_dict): template_dict['body_dur'] = body_dur -@when('it has body end time ') +@fixture +def body_end(): + return None + +@when(parsers.parse('it has body end time {body_end}')) +@when(parsers.parse('it has body end time')) def when_body_end(body_end, template_dict): template_dict['body_end'] = body_end -@when('it has div begin time ') +@fixture +def div_begin(): + return None + +@when(parsers.parse('it has div begin time {div_begin}')) +@when(parsers.parse('it has div begin time')) def when_div_begin(div_begin, template_dict): template_dict['div_begin'] = div_begin -@when('it has div end time ') +@fixture +def div_end(): + return None + +@when(parsers.parse('it has div end time {div_end}')) +@when(parsers.parse('it has div end time')) def when_div_end(div_end, template_dict): template_dict['div_end'] = div_end -@when('it has p begin time ') +@fixture +def p_begin(): + return None + +@when(parsers.parse('it has p begin time {p_begin}')) +@when(parsers.parse('it has p begin time')) def when_p_begin(p_begin, template_dict): template_dict['p_begin'] = p_begin -@when('it has p end time ') +@fixture +def p_end(): + return None + +@when(parsers.parse('it has p end time {p_end}')) +@when(parsers.parse('it has p end time')) def when_p_end(p_end, template_dict): template_dict['p_end'] = p_end -@when('it has span1 begin time ') +@fixture +def span1_begin(): + return None + +@when(parsers.parse('it has span1 begin time {span1_begin}')) +@when(parsers.parse('it has span1 begin time')) def when_span1_begin(span1_begin, template_dict): template_dict['span1_begin'] = span1_begin -@when('it has span1 end time ') +@fixture +def span1_end(): + return None + +@when(parsers.parse('it has span1 end time {span1_end}')) +@when(parsers.parse('it has span1 end time')) def when_span1_end(span1_end, template_dict): template_dict['span1_end'] = span1_end -@when('it has span2 begin time ') +@fixture +def span2_begin(): + return None + +@when(parsers.parse('it has span2 begin time {span2_begin}')) +@when(parsers.parse('it has span2 begin time')) def when_span2_begin(span2_begin, template_dict): template_dict['span2_begin'] = span2_begin -@when('it has span2 end time ') +@fixture +def span2_end(): + return None + +@when(parsers.parse('it has span2 end time {span2_end}')) +@when(parsers.parse('it has span2 end time')) def when_span2_end(span2_end, template_dict): template_dict['span2_end'] = span2_end -@when('it is available at ') +@fixture +def availability_time(): + return None + +@when(parsers.parse('it is available at {availability_time}')) +@when(parsers.parse('it is available at')) def when_doc_available(test_context, availability_time): if availability_time: test_context['document'].availability_time = FullClockTimingType(availability_time).timedelta -@then('body active begin time is ') +@then(parsers.parse('body active begin time is {body_active_begin}')) def then_body_active_begin(test_context, body_active_begin): computed_begin = test_context['document'].binding.body.computed_begin_time if body_active_begin == "undefined": @@ -92,7 +153,7 @@ def then_body_active_begin(test_context, body_active_begin): assert body_active_begin_timedelta == computed_begin -@then('body active end time is ') +@then(parsers.parse('body active end time is {body_active_end}')) def then_body_active_end(test_context, body_active_end): computed_end = test_context['document'].binding.body.computed_end_time if body_active_end == "undefined": @@ -102,7 +163,7 @@ def then_body_active_end(test_context, body_active_end): assert body_active_end_timedelta == computed_end -@then('div active begin time is ') +@then(parsers.parse('div active begin time is {div_active_begin}')) def then_div_active_begin(test_context, div_active_begin): computed_begin = test_context['document'].binding.body.orderedContent()[0].value.computed_begin_time @@ -113,7 +174,7 @@ def then_div_active_begin(test_context, div_active_begin): assert div_active_begin_timedelta == computed_begin -@then('div active end time is ') +@then(parsers.parse('div active end time is {div_active_end}')) def then_div_active_end(test_context, div_active_end): computed_end = test_context['document'].binding.body.orderedContent()[0].value.computed_end_time if div_active_end == "undefined": @@ -123,7 +184,7 @@ def then_div_active_end(test_context, div_active_end): assert div_active_end_timedelta == computed_end -@then('p active begin time is ') +@then(parsers.parse('p active begin time is {p_active_begin}')) def then_p_active_begin(test_context, p_active_begin): computed_begin = test_context['document'].binding.body.orderedContent()[0].value.orderedContent()[0].value.computed_begin_time if p_active_begin == "undefined": @@ -133,7 +194,7 @@ def then_p_active_begin(test_context, p_active_begin): assert p_active_begin_timedelta == computed_begin -@then('p active end time is ') +@then(parsers.parse('p active end time is {p_active_end}')) def then_p_active_end(test_context, p_active_end): computed_end = test_context['document'].binding.body.orderedContent()[0].value.orderedContent()[0].value.computed_end_time if p_active_end == "undefined": @@ -143,7 +204,7 @@ def then_p_active_end(test_context, p_active_end): assert p_active_end_timedelta == computed_end -@then('span1 active begin time is ') +@then(parsers.parse('span1 active begin time is {span1_active_begin}')) def then_span1_active_begin(test_context, span1_active_begin): computed_begin = test_context['document'].binding.body.orderedContent()[0].value.orderedContent()[0].value.orderedContent()[1].value.computed_begin_time if span1_active_begin == "undefined": @@ -153,7 +214,7 @@ def then_span1_active_begin(test_context, span1_active_begin): assert span1_active_begin_timedelta == computed_begin -@then('span1 active end time is ') +@then(parsers.parse('span1 active end time is {span1_active_end}')) def then_span1_active_end(test_context, span1_active_end): computed_end = test_context['document'].binding.body.orderedContent()[0].value.orderedContent()[0].value.orderedContent()[1].value.computed_end_time if span1_active_end == "undefined": @@ -163,7 +224,7 @@ def then_span1_active_end(test_context, span1_active_end): assert span1_active_end_timedelta == computed_end -@then('span2 active begin time is ') +@then(parsers.parse('span2 active begin time is {span2_active_begin}')) def then_span2_active_begin(test_context, span2_active_begin): computed_begin = test_context['document'].binding.body.orderedContent()[0].value.orderedContent()[0].value.orderedContent()[3].value.computed_begin_time if span2_active_begin == "undefined": @@ -173,7 +234,7 @@ def then_span2_active_begin(test_context, span2_active_begin): assert span2_active_begin_timedelta == computed_begin -@then('span2 active end time is ') +@then(parsers.parse('span2 active end time is {span2_active_end}')) def then_span2_active_end(test_context, span2_active_end): computed_end = test_context['document'].binding.body.orderedContent()[0].value.orderedContent()[0].value.orderedContent()[3].value.computed_end_time if span2_active_end == "undefined": diff --git a/testing/bdd/test_facet.py b/testing/bdd/test_facet.py index df3e81876..e65025fd9 100644 --- a/testing/bdd/test_facet.py +++ b/testing/bdd/test_facet.py @@ -1,10 +1,11 @@ -from pytest_bdd import when, scenarios +from pytest_bdd import when, scenarios, parsers +from pytest import fixture scenarios('features/validation/facet.feature') -@when('it has facet1 applied to element ') +@when(parsers.parse('it has facet1 applied to element {parent1}')) def when_facet1_to_parent1(parent1, template_dict, test_context): test_context['parent1'] = parent1 key = parent1 + "_facets" @@ -14,20 +15,20 @@ def when_facet1_to_parent1(parent1, template_dict, test_context): template_dict[key] = "') +@when(parsers.parse('facet1 has attribute {link1}')) def when_facet1_link1(link1, template_dict, test_context): key = test_context['parent1'] + "_facets" if link1: template_dict[key] += ' link="{}"'.format(link1) -@when('facet1 contains string ') +@when(parsers.parse('facet1 contains string {term1}')) def when_facet1_term1(term1, template_dict, test_context): key = test_context['parent1'] + "_facets" template_dict[key] += '>{}'.format(term1) -@when('it has facet2 applied to element ') +@when(parsers.parse('it has facet2 applied to element {parent2}')) def when_facet2_to_parent2(parent2, template_dict, test_context): test_context['parent2'] = parent2 key = parent2 + "_facets" @@ -37,20 +38,25 @@ def when_facet2_to_parent2(parent2, template_dict, test_context): template_dict[key] = "') +@when(parsers.parse('facet2 has attribute {link2}')) def when_facet2_link2(link2, template_dict, test_context): key = test_context['parent2'] + "_facets" if link2: template_dict[key] += ' link="{}"'.format(link2) -@when('facet2 contains string ') +@when(parsers.parse('facet2 contains string {term2}')) def when_facet2_term2(term2, template_dict, test_context): key = test_context['parent2'] + "_facets" template_dict[key] += '>{}'.format(term2) -@when('it has element facet1 with attribute ') +@fixture +def expresses1(): + return None + +@when(parsers.parse('it has element facet1 with attribute {expresses1}')) +@when(parsers.parse('it has element facet1 with attribute')) def when_facet1_expresses1(expresses1, template_dict): value = '') +@fixture +def expresses2(): + return None + +@when(parsers.parse('it has element facet2 with attribute {expresses2}')) +@when(parsers.parse('it has element facet2 with attribute')) def when_facet2_expresses2(expresses2, template_dict): value = '') +@fixture +def expresses3(): + return None + +@when(parsers.parse('it has element facet3 with attribute {expresses3}')) +@when(parsers.parse('it has element facet3 with attribute')) def when_facet3_expresses3(expresses3, template_dict): value = '') +@when(parsers.parse('documentFacet has atribute {summary}')) def when_documentFacet_summary(summary, template_dict): template_dict['document_facets'] = 'test_facet'.format(summary) diff --git a/testing/bdd/test_handover.py b/testing/bdd/test_handover.py index c583a1fb3..a0417df1e 100644 --- a/testing/bdd/test_handover.py +++ b/testing/bdd/test_handover.py @@ -1,6 +1,7 @@ +from _pytest.fixtures import fixture import pytest -from pytest_bdd import scenarios, given, when, then +from pytest_bdd import scenarios, given, when, then, parsers from ebu_tt_live.node import handover as handover_node from ebu_tt_live.documents import EBUTT3Document from ebu_tt_live.carriage.interface import IProducerCarriage @@ -11,7 +12,7 @@ scenarios('features/handover/handover.feature') -@given('a handover node with and ') +@given(parsers.parse('a handover node with {authors_group_identifier} and {sequence_identifier}'), target_fixture='given_handover_node') def given_handover_node(authors_group_identifier, sequence_identifier): carriage = MagicMock(spec=IProducerCarriage) carriage.expects.return_value = EBUTT3Document @@ -24,65 +25,75 @@ def given_handover_node(authors_group_identifier, sequence_identifier): return instance -@when('it has and ') +@when(parsers.parse('it has sequence id {sequence_identifier1} and sequence num {sequence_number1}')) def when_sequence_id_and_num_1(sequence_identifier1, sequence_number1, template_dict): template_dict['sequence_identifier'] = sequence_identifier1 template_dict['sequence_number'] = sequence_number1 -@when('it has and ') +@when(parsers.parse('new document has sequence id {sequence_identifier2} and sequence num {sequence_number2}')) def when_sequence_id_and_num_2(sequence_identifier2, sequence_number2, template_dict): template_dict['sequence_identifier'] = sequence_identifier2 template_dict['sequence_number'] = sequence_number2 -@when('it has ') +@when(parsers.parse('it has {authors_group_identifier}')) def when_authors_group_id(template_dict, authors_group_identifier): template_dict['authors_group_identifier'] = authors_group_identifier -@when('it has ') +@pytest.fixture +def authors_group_identifier1(): + return None + +@when(parsers.parse('it has authors group identifier {authors_group_identifier1}')) +@when(parsers.parse('it has authors group identifier')) def when_authors_group_id1(template_dict, authors_group_identifier1): template_dict['authors_group_identifier'] = authors_group_identifier1 -@when('it has ') +@when(parsers.parse('new document has authors group identifier {authors_group_identifier2}')) def when_authors_group_id1(template_dict, authors_group_identifier2): template_dict['authors_group_identifier'] = authors_group_identifier2 -@when('it has ') +@fixture +def authors_group_control_token1(): + return None + +@when(parsers.parse('it has authors group control token {authors_group_control_token1}')) +@when(parsers.parse('it has authors group control token')) def when_authors_group_token1(template_dict, authors_group_control_token1): template_dict['authors_group_control_token'] = authors_group_control_token1 -@when('it has ') +@when(parsers.parse('new document has authors group control token {authors_group_control_token2}')) def when_authors_group_token2(template_dict, authors_group_control_token2): template_dict['authors_group_control_token'] = authors_group_control_token2 -@when('new document is created') +@when(parsers.parse('new document is created')) def new_doc_created(template_dict): template_dict.clear() -@when('handover node processes document') +@when(parsers.parse('handover node processes document')) def new_document(test_context, given_handover_node): given_handover_node.process_document(test_context['document']) -@then('handover node emits documents') +@then(parsers.parse('handover node emits {emitted_documents} documents')) def then_handover_node_emits(given_handover_node, emitted_documents): assert given_handover_node.producer_carriage.emit_data.call_count == int(emitted_documents) -@then('handover node errors when processing document') +@then(parsers.parse('handover node errors when processing document')) def then_handover_node_errors(given_handover_node, test_context): with pytest.raises(Exception): given_handover_node.process_document(test_context['document']) -@then('the emitted documents belong to and use consecutive sequence numbering from 1') +@then(parsers.parse('the emitted documents belong to {sequence_identifier} and use consecutive sequence numbering from 1')) def then_handover_node_produces_sequence(given_handover_node, sequence_identifier): counter = 1 for pos_args, kw_args in given_handover_node.producer_carriage.emit_data.call_args_list: @@ -91,14 +102,19 @@ def then_handover_node_produces_sequence(given_handover_node, sequence_identifie counter += 1 -@then('the emitted documents have and they specify a token') +@then(parsers.parse('the emitted documents have {authors_group_identifier} and they specify a token')) def then_handover_parameter_passthrough(given_handover_node, authors_group_identifier): for pos_args, kw_args in given_handover_node.producer_carriage.emit_data.call_args_list: assert kw_args['data'].authors_group_identifier == authors_group_identifier assert kw_args['data'].authors_group_control_token is not None -@then('the emitted documents have ') +@pytest.fixture +def authors_group_selected_sequence_identifiers(): + return '' + +@then(parsers.parse('the emitted documents have {authors_group_selected_sequence_identifiers}')) +@then(parsers.parse('the emitted documents have')) def then_authors_group_selected_sequence_id(given_handover_node, authors_group_selected_sequence_identifiers): # NOTE: The comma is a valid sequenceIdentifier character but we use it as a divisor in the test. Make sure # the tests use sequence identifiers without commas. diff --git a/testing/bdd/test_padding_data_type.py b/testing/bdd/test_padding_data_type.py index 069439479..3d78b0063 100644 --- a/testing/bdd/test_padding_data_type.py +++ b/testing/bdd/test_padding_data_type.py @@ -1,10 +1,11 @@ -from pytest_bdd import when, scenarios +from pytest_bdd import when, scenarios, parsers +from pytest import fixture scenarios('features/validation/padding_data_type.feature') -@when(' has a padding attribute') +@when(parsers.parse('{tag} has a padding attribute')) def when_tag_has_padding(tag, template_dict): if tag == "tt:style": template_dict["style_padding"] = 'tts:padding="1px"' @@ -16,26 +17,46 @@ def when_tag_has_padding(tag, template_dict): template_dict["span_padding"] = 'tts:padding="1px"' -@when('it has a padding attribute') +@when(parsers.parse('it has a padding attribute')) def when_padding_attribute(template_dict): template_dict["test_padding_syntax"] = True -@when('the padding attribute has ') +@fixture +def value1(): + return '' + +@when(parsers.parse('padding attribute component 1 is {value1}')) +@when(parsers.parse('padding attribute component 1 is')) def when_padding_value1(value1, template_dict): template_dict["value1"] = value1 -@when('the padding attribute has ') +@fixture +def value2(): + return '' + +@when(parsers.parse('padding attribute component 2 is {value2}')) +@when(parsers.parse('padding attribute component 2 is')) def when_padding_value2(value2, template_dict): template_dict["value2"] = value2 -@when('the padding attribute has ') +@fixture +def value3(): + return '' + +@when(parsers.parse('padding attribute component 3 is {value3}')) +@when(parsers.parse('padding attribute component 3 is')) def when_padding_value3(value3, template_dict): template_dict["value3"] = value3 -@when('the padding attribute has ') +@fixture +def value4(): + return '' + +@when(parsers.parse('padding attribute component 4 is {value4}')) +@when(parsers.parse('padding attribute component 4 is')) def when_padding_value4(value4, template_dict): template_dict["value4"] = value4 diff --git a/testing/bdd/test_passive_node_shall_not_modify_documents.py b/testing/bdd/test_passive_node_shall_not_modify_documents.py index f26f715a1..1d14ac1c3 100644 --- a/testing/bdd/test_passive_node_shall_not_modify_documents.py +++ b/testing/bdd/test_passive_node_shall_not_modify_documents.py @@ -11,7 +11,7 @@ scenarios('features/nodes/passive_nodes_shall_not_modify_document.feature') -@given('a distributing node') +@given('a distributing node', target_fixture='distributing_node') def distributing_node(): reference_clock = MagicMock() prod_carriage = MagicMock(spec=IProducerCarriage) @@ -24,7 +24,7 @@ def distributing_node(): return node -@given('a buffer delay node') +@given('a buffer delay node', target_fixture='buffer_delay_node') def buffer_delay_node(): reference_clock = MagicMock() prod_carriage = MagicMock(spec=IProducerCarriage) diff --git a/testing/bdd/test_referenceClockIdentifier.py b/testing/bdd/test_referenceClockIdentifier.py index d79b2f857..bf8ac627e 100644 --- a/testing/bdd/test_referenceClockIdentifier.py +++ b/testing/bdd/test_referenceClockIdentifier.py @@ -1,18 +1,33 @@ -from pytest_bdd import scenarios, when +from pytest_bdd import scenarios, when, parsers +from pytest import fixture scenarios('features/validation/referenceClockIdentifier_constraints.feature') -@when('it has timeBase ') +@fixture +def clock_mode(): + return None + +@when(parsers.parse('it has timeBase {time_base}')) def when_time_base(time_base, template_dict): template_dict['time_base'] = time_base -@when('it has clock mode ') +@fixture +def clock_mode(): + return None + +@when(parsers.parse('it has clock mode {clock_mode}')) +@when(parsers.parse('it has clock mode')) def when_clock_mode(clock_mode, template_dict): template_dict['clock_mode'] = clock_mode -@when('it has reference clock identifier ') +@fixture +def ref_clock_id(): + return None + +@when(parsers.parse('it has reference clock identifier {ref_clock_id}')) +@when(parsers.parse('it has reference clock identifier')) def when_ref_clock_id(ref_clock_id, template_dict): template_dict['ref_clock_id'] = ref_clock_id diff --git a/testing/bdd/test_resolved_times.py b/testing/bdd/test_resolved_times.py index b8a0ca762..521fda8b7 100644 --- a/testing/bdd/test_resolved_times.py +++ b/testing/bdd/test_resolved_times.py @@ -1,6 +1,8 @@ from ebu_tt_live.documents import EBUTT3Document from ebu_tt_live.bindings._ebuttdt import FullClockTimingType, LimitedClockTimingType from pytest_bdd import scenarios, then, when, parsers +from pytest import fixture + scenarios('features/timing/resolved_times.feature') scenarios('features/timing/resolved_times_no_body.feature') @@ -20,12 +22,12 @@ def when_new_doc(template_dict): template_dict.clear() -@when('it has timeBase ') +@when(parsers.parse('it has timeBase {time_base}')) def when_time_base(time_base, template_dict): template_dict['time_base'] = time_base -@when('it has sequenceIdentifier ') +@when(parsers.parse('it has sequenceIdentifier {sequence_identifier}')) def when_seq_id(sequence_identifier, template_dict): template_dict['sequence_identifier'] = sequence_identifier @@ -35,67 +37,127 @@ def when_sequence_number(sequence_number, template_dict): template_dict['sequence_number'] = sequence_number -@when('it has doc1 body ') +@fixture +def doc1_no_body(): + return None + +@when(parsers.parse('it has doc1 body {doc1_no_body}')) +@when(parsers.parse('it has doc1 body')) def when_doc1_body(doc1_no_body, template_dict): template_dict['body'] = doc1_no_body -@when('it has doc1 body begin time ') +@fixture +def doc1_begin(): + return None + +@when(parsers.parse('it has doc1 body begin time {doc1_begin}')) +@when(parsers.parse('it has doc1 body begin time')) def when_doc1_body_begin(doc1_begin, template_dict): template_dict['body_begin'] = doc1_begin -@when('it has doc1 body end time ') +@fixture +def doc1_end(): + return None + +@when(parsers.parse('it has doc1 body end time {doc1_end}')) +@when(parsers.parse('it has doc1 body end time')) def when_doc1_body_end(doc1_end, template_dict): template_dict['body_end'] = doc1_end -@when('it has doc1 body duration ') +@fixture +def doc1_dur(): + return None + +@when(parsers.parse('it has doc1 body duration {doc1_dur}')) +@when(parsers.parse('it has doc1 body duration')) def when_doc1_body_dur(doc1_dur, template_dict): template_dict['body_dur'] = doc1_dur -@when('it has doc2 body ') +@fixture +def doc2_no_body(): + return None + +@when(parsers.parse('it has doc2 body {doc2_no_body}')) +@when(parsers.parse('it has doc2 body')) def when_doc2_body(doc2_no_body, template_dict): template_dict['body'] = doc2_no_body -@when('it has doc2 body begin time ') +@fixture +def doc2_begin(): + return None + +@when(parsers.parse('it has doc2 body begin time {doc2_begin}')) +@when(parsers.parse('it has doc2 body begin time')) def when_doc2_body_begin(doc2_begin, template_dict): template_dict['body_begin'] = doc2_begin -@when('it has doc2 body end time ') +@fixture +def doc2_end(): + return None + +@when(parsers.parse('it has doc2 body end time {doc2_end}')) +@when(parsers.parse('it has doc2 body end time')) def when_doc2_body_end(doc2_end, template_dict): template_dict['body_end'] = doc2_end -@when('it has doc2 body duration ') +@fixture +def doc2_dur(): + return None + +@when(parsers.parse('it has doc2 body duration {doc2_dur}')) +@when(parsers.parse('it has doc2 body duration')) def when_doc2_body_dur(doc2_dur, template_dict): template_dict['body_dur'] = doc2_dur -@when('it has doc3 body ') +@fixture +def doc3_no_body(): + return None + +@when(parsers.parse('it has doc3 body {doc3_no_body}')) +@when(parsers.parse('it has doc3 body')) def when_doc3_body(doc3_no_body, template_dict): template_dict['body'] = doc3_no_body -@when('it has doc3 body begin time ') +@fixture +def doc3_begin(): + return None + +@when(parsers.parse('it has doc3 body begin time {doc3_begin}')) +@when(parsers.parse('it has doc3 body begin time')) def when_doc3_body_begin(doc3_begin, template_dict): template_dict['body_begin'] = doc3_begin -@when('it has doc3 body end time ') +@fixture +def doc3_end(): + return None + +@when(parsers.parse('it has doc3 body end time {doc3_end}')) +@when(parsers.parse('it has doc3 body end time')) def when_doc3_body_end(doc3_end, template_dict): template_dict['body_end'] = doc3_end -@when('it has doc3 body duration ') +@fixture +def doc3_dur(): + return None + +@when(parsers.parse('it has doc3 body duration {doc3_dur}')) +@when(parsers.parse('it has doc3 body duration')) def when_doc3_body_dur(doc3_dur, template_dict): template_dict['body_dur'] = doc3_dur -@when('doc1 is added to the sequence with availability time ') +@when(parsers.parse('doc1 is added to the sequence with availability time {doc1_avail_time}')) def when_doc1_added_to_seq(doc1_avail_time, template_file, template_dict, sequence, test_context): xml_file = template_file.render(template_dict) document = EBUTT3Document.create_from_xml(xml_file) @@ -104,7 +166,7 @@ def when_doc1_added_to_seq(doc1_avail_time, template_file, template_dict, sequen test_context['doc1'] = document -@when('doc2 is added to the sequence with availability time ') +@when(parsers.parse('doc2 is added to the sequence with availability time {doc2_avail_time}')) def when_doc2_added_to_seq(doc2_avail_time, template_file, template_dict, sequence, test_context): xml_file = template_file.render(template_dict) document = EBUTT3Document.create_from_xml(xml_file) @@ -113,7 +175,7 @@ def when_doc2_added_to_seq(doc2_avail_time, template_file, template_dict, sequen test_context['doc2'] = document -@when('doc3 is added to the sequence with availability time ') +@when(parsers.parse('doc3 is added to the sequence with availability time {doc3_avail_time}')) def when_doc3_added_to_seq(doc3_avail_time, template_file, template_dict, sequence, test_context): xml_file = template_file.render(template_dict) document = EBUTT3Document.create_from_xml(xml_file) @@ -122,13 +184,18 @@ def when_doc3_added_to_seq(doc3_avail_time, template_file, template_dict, sequen test_context['doc3'] = document -@then('doc1 has resolved begin time ') +@then(parsers.parse('doc1 has resolved begin time {r_begin_doc1}')) def valid_resolved_begin_time_doc1(r_begin_doc1, sequence, test_context): resolved_begin_timedelta = timestr_to_timedelta(r_begin_doc1, sequence.reference_clock.time_base) assert test_context['doc1'].resolved_begin_time == resolved_begin_timedelta -@then('doc1 has resolved end time ') +@fixture +def r_end_doc1(): + return None + +@then(parsers.parse('doc1 has resolved end time {r_end_doc1}')) +@then(parsers.parse('doc1 has resolved end time')) def valid_resolved_end_time_doc1(r_end_doc1, sequence, test_context): if r_end_doc1: resolved_end_timedelta = timestr_to_timedelta(r_end_doc1, sequence.reference_clock.time_base) @@ -137,13 +204,13 @@ def valid_resolved_end_time_doc1(r_end_doc1, sequence, test_context): assert test_context['doc1'].resolved_end_time == resolved_end_timedelta -@then('doc2 has resolved begin time ') +@then(parsers.parse('doc2 has resolved begin time {r_begin_doc2}')) def valid_resolved_begin_time_doc2(r_begin_doc2, sequence, test_context): resolved_begin_timedelta = timestr_to_timedelta(r_begin_doc2, sequence.reference_clock.time_base) assert test_context['doc2'].resolved_begin_time == resolved_begin_timedelta -@then('doc2 has resolved end time ') +@then(parsers.parse('doc2 has resolved end time {r_end_doc2}')) def valid_resolved_end_time_doc2(r_end_doc2, sequence, test_context): if r_end_doc2: resolved_end_timedelta = timestr_to_timedelta(r_end_doc2, sequence.reference_clock.time_base) @@ -152,13 +219,18 @@ def valid_resolved_end_time_doc2(r_end_doc2, sequence, test_context): assert test_context['doc2'].resolved_end_time == resolved_end_timedelta -@then('doc3 has resolved begin time ') +@then(parsers.parse('doc3 has resolved begin time {r_begin_doc3}')) def valid_resolved_begin_time_doc3(r_begin_doc3, sequence, test_context): resolved_begin_timedelta = timestr_to_timedelta(r_begin_doc3, sequence.reference_clock.time_base) assert test_context['doc3'].resolved_begin_time == resolved_begin_timedelta -@then('doc3 has resolved end time ') +@fixture +def r_end_doc3(): + return None + +@then(parsers.parse('doc3 has resolved end time {r_end_doc3}')) +@then(parsers.parse('doc3 has resolved end time')) def valid_resolved_end_time_doc3(r_end_doc3, sequence, test_context): if r_end_doc3: resolved_end_timedelta = timestr_to_timedelta(r_end_doc3, sequence.reference_clock.time_base) diff --git a/testing/bdd/test_time_regex_parsing.py b/testing/bdd/test_time_regex_parsing.py index 6b014bff7..bf94dfaf5 100644 --- a/testing/bdd/test_time_regex_parsing.py +++ b/testing/bdd/test_time_regex_parsing.py @@ -1,8 +1,8 @@ from ebu_tt_live.documents import EBUTT3Document from datetime import timedelta -from pytest_bdd import scenarios, when, then +from pytest_bdd import scenarios, when, then, parsers -scenarios('features/validation/time_regex_parsing.feature', example_converters=dict(trusted_timedeltas_index=int)) +scenarios('features/validation/time_regex_parsing.feature') trusted_timedeltas = [ timedelta(hours=15), @@ -24,7 +24,7 @@ def when_body_begin(body_begin, template_dict): template_dict['body_begin'] = body_begin -@then('timedelta value given when reading body.begin should be ') +@then(parsers.parse('timedelta value given when reading body.begin should be {trusted_timedeltas_index:d}')) def check_correct_parsing(template_file, template_dict, trusted_timedeltas_index): xml_file = template_file.render(template_dict) document = EBUTT3Document.create_from_xml(xml_file) From 51bc3dfe8a5d8c654750aac35294b15792e7ac3c Mon Sep 17 00:00:00 2001 From: Nigel Megitt Date: Mon, 6 Dec 2021 15:44:36 +0000 Subject: [PATCH 6/7] Fix up the remaining BDD tests --- testing/bdd/conftest.py | 5 + .../features/timing/retimingDelayNode.feature | 10 +- .../sequence_identical_timing_model.feature | 28 +- testing/bdd/test_retimingDelayNode.py | 298 ++++++++++++++---- testing/bdd/test_segmentation.py | 74 +++-- .../test_sequence_attributes_validation.py | 18 +- .../test_sequence_identical_timing_model.py | 93 +++++- testing/bdd/test_smpte.py | 38 ++- testing/bdd/test_style.py | 81 ++++- .../bdd/test_timeBase_attribute_mandatory.py | 11 +- ...test_timeBase_clock_clockMode_mandatory.py | 12 +- testing/bdd/test_timeBase_timeformat.py | 69 +++- testing/bdd/test_time_regex_parsing.py | 4 +- testing/bdd/test_websocket_carriage_config.py | 10 +- testing/bdd/test_xml_lang_attribute.py | 10 +- 15 files changed, 585 insertions(+), 176 deletions(-) diff --git a/testing/bdd/conftest.py b/testing/bdd/conftest.py index f79a3d85f..f3de5a2fd 100644 --- a/testing/bdd/conftest.py +++ b/testing/bdd/conftest.py @@ -177,7 +177,12 @@ def valid_computed_end_time(computed_end, gen_document): } +@pytest.fixture +def computed_value(): + return '' + @then(parsers.parse('the computed {style_attribute} in {elem_id} is {computed_value}')) +@then(parsers.parse('the computed {style_attribute} in {elem_id} is')) def then_computed_style_value_is(style_attribute, elem_id, computed_value, test_context): document = test_context['document'] elem = document.get_element_by_id(elem_id) diff --git a/testing/bdd/features/timing/retimingDelayNode.feature b/testing/bdd/features/timing/retimingDelayNode.feature index bd765797c..fbe674496 100644 --- a/testing/bdd/features/timing/retimingDelayNode.feature +++ b/testing/bdd/features/timing/retimingDelayNode.feature @@ -114,12 +114,12 @@ Feature: Delay of a document sequence # SPEC-CONFORMANCE.md R113 R117 Scenario: RetimingDelayNode changes sequence ID but not authoring delay Given an xml file - And it has - And it has + And it has sequence id + And it has authoring delay And the document is generated When the retiming delay node delays it by - Then the updated document has - And the updated document has + Then the updated document has sequence id + And the updated document has authoring delay Examples: | sequence_id_1 | authoring_delay | delay | sequence_id_2 | @@ -130,7 +130,7 @@ Feature: Delay of a document sequence # The above scenario brings up a problem with the incoming sequence identifier matching the produced one Scenario: Retiming delay receives matching sequence identifier Given an xml file - And it has + And it has sequence id And the document is generated Then the retiming delay node with will reject it diff --git a/testing/bdd/features/validation/sequence_identical_timing_model.feature b/testing/bdd/features/validation/sequence_identical_timing_model.feature index 1c483b5e4..d16860350 100644 --- a/testing/bdd/features/validation/sequence_identical_timing_model.feature +++ b/testing/bdd/features/validation/sequence_identical_timing_model.feature @@ -23,13 +23,13 @@ Feature: Every document in a sequence shall have an identical timing model as de And it has markerMode And doc1 is added to the sequence And we create a new document - And it has sequenceNumber 2 - And it has timeBase - And it has clockMode - And it has frameRate - And it has frameRateMultiplier - And it has dropMode - And it has markerMode + And it has sequenceNumber2 2 + And it has timeBase2 + And it has clockMode2 + And it has frameRate2 + And it has frameRateMultiplier2 + And it has dropMode2 + And it has markerMode2 Then adding doc2 to the sequence results in an error Examples: @@ -59,13 +59,13 @@ Feature: Every document in a sequence shall have an identical timing model as de And it has markerMode And doc1 is added to the sequence And we create a new document - And it has sequenceNumber 2 - And it has timeBase - And it has clockMode - And it has frameRate - And it has frameRateMultiplier - And it has dropMode - And it has markerMode + And it has sequenceNumber2 2 + And it has timeBase2 + And it has clockMode2 + And it has frameRate2 + And it has frameRateMultiplier2 + And it has dropMode2 + And it has markerMode2 Then adding doc2 to the sequence does not raise any error Examples: diff --git a/testing/bdd/test_retimingDelayNode.py b/testing/bdd/test_retimingDelayNode.py index ef17ac207..b1321f0ab 100644 --- a/testing/bdd/test_retimingDelayNode.py +++ b/testing/bdd/test_retimingDelayNode.py @@ -5,78 +5,137 @@ from ebu_tt_live.documents import EBUTT3Document from ebu_tt_live.errors import UnexpectedSequenceIdentifierError from mock import MagicMock -from pytest_bdd import scenarios, given, when, then +from pytest_bdd import scenarios, given, when, then, parsers import pytest scenarios('features/timing/retimingDelayNode.feature') +@pytest.fixture +def body_begin(): + return None -@given('it has body begin time ') +@given(parsers.parse('it has body begin time {body_begin}'), target_fixture='given_body_begin') +@given(parsers.parse('it has body begin time'), target_fixture='given_body_begin') def given_body_begin(body_begin, template_dict): template_dict['body_begin'] = body_begin -@given('it has body end time ') +@pytest.fixture +def body_end(): + return None + +@given(parsers.parse('it has body end time {body_end}'), target_fixture='given_body_end') +@given(parsers.parse('it has body end time'), target_fixture='given_body_end') def given_body_end(body_end, template_dict): template_dict['body_end'] = body_end -@given('it has body duration ') +@pytest.fixture +def body_dur(): + return None + +@given(parsers.parse('it has body duration {body_dur}'), target_fixture='given_body_dur') +@given(parsers.parse('it has body duration'), target_fixture='given_body_dur') def given_body_dur(body_dur, template_dict): template_dict['body_dur'] = body_dur -@given('it has div begin time ') +@pytest.fixture +def div_begin(): + return None + +@given(parsers.parse('it has div begin time {div_begin}'), target_fixture='given_div_begin') +@given(parsers.parse('it has div begin time'), target_fixture='given_div_begin') def given_div_begin(div_begin, template_dict): template_dict['div_begin'] = div_begin -@given('it has div end time ') +@pytest.fixture +def div_end(): + return None + +@given(parsers.parse('it has div end time {div_end}'), target_fixture='given_div_end') +@given(parsers.parse('it has div end time'), target_fixture='given_div_end') def given_div_end(div_end, template_dict): template_dict['div_end'] = div_end -@given('it has p begin time ') +@pytest.fixture +def p_begin(): + return None + +@given(parsers.parse('it has p begin time {p_begin}'), target_fixture='given_p_begin') +@given(parsers.parse('it has p begin time'), target_fixture='given_p_begin') def given_p_begin(p_begin, template_dict): template_dict['p_begin'] = p_begin -@given('it has p end time ') +@pytest.fixture +def p_end(): + return None + +@given(parsers.parse('it has p end time {p_end}'), target_fixture='given_p_end') +@given(parsers.parse('it has p end time'), target_fixture='given_p_end') def given_p_end(p_end, template_dict): template_dict['p_end'] = p_end -@given('it has span begin time ') +@pytest.fixture +def span_begin(): + return None + +@given(parsers.parse('it has span begin time {span_begin}'), target_fixture='given_span_begin') +@given(parsers.parse('it has span begin time'), target_fixture='given_span_begin') def given_span_begin(span_begin, template_dict): template_dict['span_begin'] = span_begin -@given('it has span end time ') +@pytest.fixture +def span_end(): + return None + +@given(parsers.parse('it has span end time {span_end}'), target_fixture='given_span_end') +@given(parsers.parse('it has span end time'), target_fixture='given_span_end') def given_span_end(span_end, template_dict): template_dict['span_end'] = span_end -@given('it has span2 begin time ') +@pytest.fixture +def span2_begin(): + return None + +@given(parsers.parse('it has span2 begin time {span2_begin}'), target_fixture='given_span2_begin') +@given(parsers.parse('it has span2 begin time'), target_fixture='given_span2_begin') def given_span2_begin(span2_begin, template_dict): template_dict['span2_begin'] = span2_begin -@given('it has span2 end time ') +@pytest.fixture +def span2_end(): + return None + +@given(parsers.parse('it has span2 end time {span2_end}'), target_fixture='given_span2_end') +@given(parsers.parse('it has span2 end time'), target_fixture='given_span2_end') def given_span2_end(span2_end, template_dict): template_dict['span2_end'] = span2_end -@given('it has ') +@given(parsers.parse('it has sequence id {sequence_id_1}'), target_fixture='given_original_sequence_id') def given_original_sequence_id(template_dict, sequence_id_1): template_dict['sequence_identifier'] = sequence_id_1 -@given('it has ') +@pytest.fixture +def authoring_delay(): + return None + +@given(parsers.parse('it has authoring delay {authoring_delay}'), target_fixture='given_authoring_delay') +@given(parsers.parse('it has authoring delay'), target_fixture='given_authoring_delay') def given_authoring_delay(template_dict, authoring_delay): template_dict['authoring_delay'] = authoring_delay -@when('the retiming delay node delays it by ') +@when(parsers.parse('the retiming delay node delays it by {delay}')) def when_retiming_delay(delay, test_context, gen_document): reference_clock = LocalMachineClock() @@ -99,7 +158,7 @@ def when_retiming_delay(delay, test_context, gen_document): test_context['doc'] = gen_document -@then('the retiming delay node with will reject it') +@then(parsers.parse('the retiming delay node with {produced_sequence} will reject it')) def then_retiming_delay_node_rejects(gen_document, produced_sequence): reference_clock = LocalMachineClock() reference_clock.clock_mode = 'local' @@ -118,7 +177,7 @@ def then_retiming_delay_node_rejects(gen_document, produced_sequence): delay_node.process_document(gen_document) -@then('the delay node outputs the document at ') +@then(parsers.parse('the delay node outputs the document at {delayed_avail_time}')) def then_availability_time(delayed_avail_time, test_context): delayed_avail_time_float = LimitedClockTimingType(delayed_avail_time).timedelta assert test_context['doc'].availability_time == delayed_avail_time_float @@ -126,7 +185,12 @@ def then_availability_time(delayed_avail_time, test_context): # functions for computed times -@then('the updated body computed begin time is ') +@pytest.fixture +def updated_body_begin(): + return '' + +@then(parsers.parse('the updated body computed begin time is {updated_body_begin}')) +@then(parsers.parse('the updated body computed begin time is')) def then_updated_body_computed_begin_time(test_context, updated_body_begin): if updated_body_begin != '': assert test_context['doc'].binding.body.computed_begin_time == LimitedClockTimingType(updated_body_begin).timedelta @@ -134,7 +198,12 @@ def then_updated_body_computed_begin_time(test_context, updated_body_begin): assert test_context['doc'].binding.body.computed_begin_time is None -@then('the updated body computed end time is ') +@pytest.fixture +def updated_body_end(): + return '' + +@then(parsers.parse('the updated body computed end time is {updated_body_end}')) +@then(parsers.parse('the updated body computed end time is')) def then_updated_body_computed_end_time(test_context, updated_body_end): if updated_body_end != '': assert test_context['doc'].binding.body.computed_end_time == LimitedClockTimingType(updated_body_end).timedelta @@ -142,7 +211,12 @@ def then_updated_body_computed_end_time(test_context, updated_body_end): assert test_context['doc'].binding.body.computed_end_time is None -@then('the updated div computed begin time is ') +@pytest.fixture +def updated_div_begin(): + return '' + +@then(parsers.parse('the updated div computed begin time is {updated_div_begin}')) +@then(parsers.parse('the updated div computed begin time is')) def then_updated_div_computed_begin_time(test_context, updated_div_begin): if updated_div_begin != '': assert test_context['doc'].binding.body.div[0].computed_begin_time == LimitedClockTimingType(updated_div_begin).timedelta @@ -150,7 +224,12 @@ def then_updated_div_computed_begin_time(test_context, updated_div_begin): assert test_context['doc'].binding.body.div[0].computed_begin_time is None -@then('the updated div computed end time is ') +@pytest.fixture +def updated_div_end(): + return '' + +@then(parsers.parse('the updated div computed end time is {updated_div_end}')) +@then(parsers.parse('the updated div computed end time is')) def then_updated_div_computed_end_time(test_context, updated_div_end): if updated_div_end != '': assert test_context['doc'].binding.body.div[0].computed_end_time == LimitedClockTimingType(updated_div_end).timedelta @@ -158,7 +237,12 @@ def then_updated_div_computed_end_time(test_context, updated_div_end): assert test_context['doc'].binding.body.div[0].computed_end_time is None -@then('the updated p computed begin time is ') +@pytest.fixture +def updated_p_begin(): + return '' + +@then(parsers.parse('the updated p computed begin time is {updated_p_begin}')) +@then(parsers.parse('the updated p computed begin time is')) def then_updated_p_computed_begin_time(test_context, updated_p_begin): if updated_p_begin != '': assert test_context['doc'].binding.body.div[0].p[0].computed_begin_time == LimitedClockTimingType(updated_p_begin).timedelta @@ -166,7 +250,12 @@ def then_updated_p_computed_begin_time(test_context, updated_p_begin): assert test_context['doc'].binding.body.div[0].p[0].computed_begin_time is None -@then('the updated p computed end time is ') +@pytest.fixture +def updated_p_end(): + return '' + +@then(parsers.parse('the updated p computed end time is {updated_p_end}')) +@then(parsers.parse('the updated p computed end time is')) def then_updated_p_computed_end_time(test_context, updated_p_end): if updated_p_end != '': assert test_context['doc'].binding.body.div[0].p[0].computed_end_time == LimitedClockTimingType(updated_p_end).timedelta @@ -174,7 +263,12 @@ def then_updated_p_computed_end_time(test_context, updated_p_end): assert test_context['doc'].binding.body.div[0].p[0].computed_end_time is None -@then('the updated span computed begin time is ') +@pytest.fixture +def updated_span_begin(): + return '' + +@then(parsers.parse('the updated span computed begin time is {updated_span_begin}')) +@then(parsers.parse('the updated span computed begin time is')) def then_updated_span_computed_begin_time(test_context, updated_span_begin): if updated_span_begin != '': assert test_context['doc'].binding.body.div[0].p[0].span[0].computed_begin_time == LimitedClockTimingType(updated_span_begin).timedelta @@ -182,7 +276,12 @@ def then_updated_span_computed_begin_time(test_context, updated_span_begin): assert test_context['doc'].binding.body.div[0].p[0].span[0].computed_begin_time is None -@then('the updated span computed end time is ') +@pytest.fixture +def updated_span_end(): + return '' + +@then(parsers.parse('the updated span computed end time is {updated_span_end}')) +@then(parsers.parse('the updated span computed end time is')) def then_updated_span_computed_end_time(test_context, updated_span_end): if updated_span_end != '': assert test_context['doc'].binding.body.div[0].p[0].span[0].computed_end_time == LimitedClockTimingType(updated_span_end).timedelta @@ -190,7 +289,12 @@ def then_updated_span_computed_end_time(test_context, updated_span_end): assert test_context['doc'].binding.body.div[0].p[0].span[0].computed_end_time is None -@then('the updated span2 computed begin time is ') +@pytest.fixture +def updated_span2_begin(): + return '' + +@then(parsers.parse('the updated span2 computed begin time is {updated_span2_begin}')) +@then(parsers.parse('the updated span2 computed begin time is')) def then_updated_span2_computed_begin_time(test_context, updated_span2_begin): if updated_span2_begin != '': assert test_context['doc'].binding.body.div[0].p[0].span[1].computed_begin_time == LimitedClockTimingType(updated_span2_begin).timedelta @@ -198,7 +302,12 @@ def then_updated_span2_computed_begin_time(test_context, updated_span2_begin): assert test_context['doc'].binding.body.div[0].p[0].span[1].computed_begin_time is None -@then('the updated span2 computed end time is ') +@pytest.fixture +def updated_span2_end(): + return '' + +@then(parsers.parse('the updated span2 computed end time is {updated_span2_end}')) +@then(parsers.parse('the updated span2 computed end time is')) def then_updated_span2_computed_end_time(test_context, updated_span2_end): if updated_span2_end != '': assert test_context['doc'].binding.body.div[0].p[0].span[1].computed_end_time == LimitedClockTimingType(updated_span2_end).timedelta @@ -208,39 +317,64 @@ def then_updated_span2_computed_end_time(test_context, updated_span2_end): # functions for specified times -@then('the updated body specified begin time is ') -def then_updated_body_specified_begin_time(test_context, updated_body_begin): - if updated_body_begin != '': - assert test_context['doc'].binding.body.begin == updated_body_begin +@pytest.fixture +def updated_body_specified_begin(): + return '' + +@then(parsers.parse('the updated body specified begin time is {updated_body_specified_begin}')) +@then(parsers.parse('the updated body specified begin time is')) +def then_updated_body_specified_begin_time(test_context, updated_body_specified_begin): + if updated_body_specified_begin != '': + assert test_context['doc'].binding.body.begin == updated_body_specified_begin else: assert test_context['doc'].binding.body.begin is None -@then('the updated body specified end time is ') -def then_updated_body_specified_end_time(test_context, updated_body_end): - if updated_body_end != '': - assert test_context['doc'].binding.body.end == updated_body_end +@pytest.fixture +def updated_body_specified_end(): + return '' + +@then(parsers.parse('the updated body specified end time is {updated_body_specified_end}')) +@then(parsers.parse('the updated body specified end time is')) +def then_updated_body_specified_end_time(test_context, updated_body_specified_end): + if updated_body_specified_end != '': + assert test_context['doc'].binding.body.end == updated_body_specified_end else: assert test_context['doc'].binding.body.end is None -@then('the updated div specified begin time is ') -def then_updated_div_specified_begin_time(test_context, updated_div_begin): - if updated_div_begin != '': - assert test_context['doc'].binding.body.div[0].begin == updated_div_begin +@pytest.fixture +def updated_div_specified_begin(): + return '' + +@then(parsers.parse('the updated div specified begin time is {updated_div_specified_begin}')) +@then(parsers.parse('the updated div specified begin time is')) +def then_updated_div_specified_begin_time(test_context, updated_div_specified_begin): + if updated_div_specified_begin != '': + assert test_context['doc'].binding.body.div[0].begin == updated_div_specified_begin else: assert test_context['doc'].binding.body.div[0].begin is None -@then('the updated div specified end time is ') -def then_updated_div_specified_end_time(test_context, updated_div_end): - if updated_div_end != '': - assert test_context['doc'].binding.body.div[0].end == updated_div_end +@pytest.fixture +def updated_div_specified_end(): + return '' + +@then(parsers.parse('the updated div specified end time is {updated_div_specified_end}')) +@then(parsers.parse('the updated div specified end time is')) +def then_updated_div_specified_end_time(test_context, updated_div_specified_end): + if updated_div_specified_end != '': + assert test_context['doc'].binding.body.div[0].end == updated_div_specified_end else: assert test_context['doc'].binding.body.div[0].end is None -@then('the updated p specified begin time is ') +@pytest.fixture +def updated_p_begin(): + return '' + +@then(parsers.parse('the updated p specified begin time is {updated_p_begin}')) +@then(parsers.parse('the updated p specified begin time is')) def then_updated_p_specified_begin_time(test_context, updated_p_begin): if updated_p_begin != '': assert test_context['doc'].binding.body.div[0].p[0].begin == updated_p_begin @@ -248,52 +382,84 @@ def then_updated_p_specified_begin_time(test_context, updated_p_begin): assert test_context['doc'].binding.body.div[0].p[0].begin is None -@then('the updated p specified end time is ') -def then_updated_p_specified_end_time(test_context, updated_p_end): - if updated_p_end != '': - assert test_context['doc'].binding.body.div[0].p[0].end == updated_p_end +@pytest.fixture +def updated_p_specified_end(): + return '' + +@then(parsers.parse('the updated p specified end time is {updated_p_specified_end}')) +@then(parsers.parse('the updated p specified end time is')) +def then_updated_p_specified_end_time(test_context, updated_p_specified_end): + if updated_p_specified_end != '': + assert test_context['doc'].binding.body.div[0].p[0].end == updated_p_specified_end else: assert test_context['doc'].binding.body.div[0].p[0].end is None -@then('the updated span specified begin time is ') -def then_updated_span_specified_begin_time(test_context, updated_span_begin): - if updated_span_begin != '': - assert test_context['doc'].binding.body.div[0].p[0].span[0].begin == updated_span_begin +@pytest.fixture +def updated_span_specified_begin(): + return '' + +@then(parsers.parse('the updated span specified begin time is {updated_span_specified_begin}')) +@then(parsers.parse('the updated span specified begin time is')) +def then_updated_span_specified_begin_time(test_context, updated_span_specified_begin): + if updated_span_specified_begin != '': + assert test_context['doc'].binding.body.div[0].p[0].span[0].begin == updated_span_specified_begin else: assert test_context['doc'].binding.body.div[0].p[0].span[0].begin is None -@then('the updated span specified end time is ') -def then_updated_span_specified_end_time(test_context, updated_span_end): - if updated_span_end != '': - assert test_context['doc'].binding.body.div[0].p[0].span[0].end == updated_span_end +@pytest.fixture +def updated_span_specified_end(): + return '' + +@then(parsers.parse('the updated span specified end time is {updated_span_specified_end}')) +@then(parsers.parse('the updated span specified end time is')) +def then_updated_span_specified_end_time(test_context, updated_span_specified_end): + if updated_span_specified_end != '': + assert test_context['doc'].binding.body.div[0].p[0].span[0].end == updated_span_specified_end else: assert test_context['doc'].binding.body.div[0].p[0].span[0].end is None -@then('the updated span2 specified begin time is ') -def then_updated_span2_specified_begin_time(test_context, updated_span2_begin): - if updated_span2_begin != '': - assert test_context['doc'].binding.body.div[0].p[0].span[1].begin == updated_span2_begin +@pytest.fixture +def updated_span2_specified_begin(): + return '' + +@then(parsers.parse('the updated span2 specified begin time is {updated_span2_specified_begin}')) +@then(parsers.parse('the updated span2 specified begin time is')) +def then_updated_span2_specified_begin_time(test_context, updated_span2_specified_begin): + if updated_span2_specified_begin != '': + assert test_context['doc'].binding.body.div[0].p[0].span[1].begin == updated_span2_specified_begin else: assert test_context['doc'].binding.body.div[0].p[0].span[1].begin is None -@then('the updated span2 specified end time is ') -def then_updated_span2_specified_end_time(test_context, updated_span2_end): - if updated_span2_end != '': - assert test_context['doc'].binding.body.div[0].p[0].span[1].end == updated_span2_end +@pytest.fixture +def updated_span2_specified_end(): + return '' + +@then(parsers.parse('the updated span2 specified end time is {updated_span2_specified_end}')) +@then(parsers.parse('the updated span2 specified end time is')) +def then_updated_span2_specified_end_time(test_context, updated_span2_specified_end): + if updated_span2_specified_end != '': + assert test_context['doc'].binding.body.div[0].p[0].span[1].end == updated_span2_specified_end else: assert test_context['doc'].binding.body.div[0].p[0].span[1].end is None -@then('the updated document has ') +@pytest.fixture +def sequence_id_2(): + return None + +@then(parsers.parse('the updated document has sequence id {sequence_id_2}')) +@then(parsers.parse('the updated document has sequence id')) def then_updated_seq_id(test_context, sequence_id_2): assert test_context['doc'].sequence_identifier == sequence_id_2 -@then('the updated document has ') +# null authoring_delay fixture defined for previous step earlier in this file +@then(parsers.parse('the updated document has authoring delay {authoring_delay}')) +@then(parsers.parse('the updated document has authoring delay')) def then_updated_auth_delay(test_context, authoring_delay): if authoring_delay: assert test_context['doc'].binding.authoringDelay == authoring_delay diff --git a/testing/bdd/test_segmentation.py b/testing/bdd/test_segmentation.py index d10aaa744..02e02f8fc 100644 --- a/testing/bdd/test_segmentation.py +++ b/testing/bdd/test_segmentation.py @@ -1,10 +1,11 @@ +import pytest from ebu_tt_live.documents import EBUTT3Document from ebu_tt_live.bindings import style_type, region_type from ebu_tt_live.bindings._ebuttdt import FullClockTimingType from ebu_tt_live.node.distributing import DistributingNode from ebu_tt_live.carriage.interface import IProducerCarriage, IConsumerCarriage from mock import MagicMock -from pytest_bdd import scenarios, when, then, given +from pytest_bdd import scenarios, when, then, given, parsers import six scenarios('features/segmentation/splitting_documents.feature') @@ -19,7 +20,7 @@ def assert_raises(exc_class, callable, *args, **kwargs): assert isinstance(exc, exc_class) -@when('we create a new document with ') +@when(parsers.parse('we create a new document with {body_dur} {span1_begin} {span2_begin} {span1_end} {span2_end}')) def when_new_doc(template_dict, sequence, body_dur, span1_begin, span1_end, span2_begin, span2_end): template_dict.clear() new_dummy_doc = sequence.new_document() @@ -40,27 +41,27 @@ def when_doc_added_to_sequence(template_file, template_dict, sequence): sequence.add_document(document) -@when('it has sequenceIdentifier ') +@when(parsers.parse('it has sequenceIdentifier {sequence_identifier}')) def when_sequence_identifier(template_dict, sequence_identifier): template_dict['sequence_identifier'] = sequence_identifier -@when('it has sequence identifier ') +@when(parsers.parse('it has sequence identifier {seq_id_1}')) def when_seq_id_1(seq_id_1, template_dict): template_dict['sequence_id'] = seq_id_1 -@when('it has sequence identifier ') +@when(parsers.parse('it has sequence identifier {seq_id_2}')) def when_seq_id_2(seq_id_2, template_dict): template_dict['sequence_id'] = seq_id_2 -@when('it has sequence number ') +@when(parsers.parse('it has sequence number {seq_n_1}')) def when_seq_num_1(template_dict, seq_n_1): template_dict['sequence_num'] = seq_n_1 -@when('it has sequence number ') +@when(parsers.parse('it has sequence number {seq_n_2}')) def when_seq_num_2(template_dict, seq_n_2): template_dict['sequence_num'] = seq_n_2 @@ -70,7 +71,7 @@ def when_another_document_arrives(template_dict): template_dict.clear() -@given('a processing node') +@given('a processing node', target_fixture='given_processing_node') def given_processing_node(template_dict): producer_carriage = MagicMock(spec=IProducerCarriage) producer_carriage.expects.return_value = six.text_type @@ -95,12 +96,12 @@ def when_document_processed(test_context, given_processing_node): given_processing_node.producer_carriage.reset_mock() -@when('the document has availability time ') +@when(parsers.parse('the document has availability time {avail_time_1}')) def when_document_availability_time(test_context, avail_time_1): test_context['document'].availability_time = FullClockTimingType(avail_time_1).timedelta -@then('the document has availability time ') +@then(parsers.parse('the document has availability time {avail_time_1}')) def then_document_availability_time(test_context, avail_time_1): assert test_context['document'].availability_time == FullClockTimingType(avail_time_1).timedelta @@ -112,51 +113,51 @@ def then_document_not_processed(test_context, given_processing_node): given_processing_node.producer_carriage.reset_mock() -@when('it has sequenceNumber ') +@when(parsers.parse('it has sequenceNumber {sequence_number}')) def when_sequence_number(template_dict, sequence_number): template_dict['sequence_number'] = sequence_number -@when('it has body from to ') +@when(parsers.parse('it has body from {body_begin} to {body_end}')) def when_body_times(template_dict, body_begin, body_end): template_dict['body_begin'] = body_begin template_dict['body_end'] = body_end -@when('body begins at ') +@when(parsers.parse('body begins at {body1_begin}')) def when_body1_begins(template_dict, body1_begin): template_dict['body_begin'] = body1_begin -@when('body begins at ') +@when(parsers.parse('body begins at {body2_begin}')) def when_body2_begins(template_dict, body2_begin): template_dict['body_begin'] = body2_begin -@when('body begins at ') +@when(parsers.parse('body begins at {body3_begin}')) def when_body3_begins(template_dict, body3_begin): template_dict['body_begin'] = body3_begin -@when('it has span1 from to ') +@when(parsers.parse('it has span1 from {span1_begin} to {span1_end}')) def when_span1_times(template_dict, span1_begin, span1_end): template_dict['span1_begin'] = span1_begin template_dict['span1_end'] = span1_end -@when('it has span2 from to ') +@when(parsers.parse('it has span2 from {span2_begin} to {span2_end}')) def when_span2_times(template_dict, span2_begin, span2_end): template_dict['span2_begin'] = span2_begin template_dict['span2_end'] = span2_end -@when('it has span3 from to ') +@when(parsers.parse('it has span3 from {span3_begin} to {span3_end}')) def when_span3_times(template_dict, span3_begin, span3_end): template_dict['span3_begin'] = span3_begin template_dict['span3_end'] = span3_end -@when('the range from to is requested') +@when(parsers.parse('the range from {range_from} to {range_to} is requested')) def when_range_requested(template_file, test_context, template_dict, range_from, range_to): xml_file = template_file.render(template_dict) document = EBUTT3Document.create_from_xml(xml_file) @@ -167,7 +168,7 @@ def when_range_requested(template_file, test_context, template_dict, range_from, test_context['fragment'] = fragment -@when('the sequence is segmented from to ') +@when(parsers.parse('the sequence is segmented from {range_from} to {range_to}')) def when_sequence_segmented(sequence, test_context, range_from, range_to): fragment = sequence.extract_segment( FullClockTimingType(range_from).timedelta, @@ -176,7 +177,7 @@ def when_sequence_segmented(sequence, test_context, range_from, range_to): test_context['fragment'] = fragment -@then('the fragment contains body with computed times from to ') +@then(parsers.parse('the fragment contains body with computed times from {frag_body_begin} to {frag_body_end}')) def then_fragment_body_times(test_context, frag_body_begin, frag_body_end): assert test_context['fragment'].binding.body.computed_begin_time == FullClockTimingType(frag_body_begin).timedelta @@ -186,7 +187,12 @@ def then_fragment_body_times(test_context, frag_body_begin, frag_body_end): assert test_context['fragment'].binding.body.computed_end_time == FullClockTimingType(frag_body_end).timedelta -@then('the fragment contains span1 with computed times from to ') +@pytest.fixture +def frag_span1_end(): + return '' + +@then(parsers.parse('the fragment contains span1 with computed times from {frag_span1_begin} to {frag_span1_end}')) +@then(parsers.parse('the fragment contains span1 with computed times from {frag_span1_begin} to')) def then_fragment_span1_times(test_context, frag_span1_begin, frag_span1_end): if frag_span1_begin == 'deleted': assert_raises(LookupError, test_context['fragment'].get_element_by_id, 'span1') @@ -200,7 +206,12 @@ def then_fragment_span1_times(test_context, frag_span1_begin, frag_span1_end): frag_span1_end).timedelta -@then('the fragment contains span2 with computed times from to ') +@pytest.fixture +def frag_span2_end(): + return '' + +@then(parsers.parse('the fragment contains span2 with computed times from {frag_span2_begin} to {frag_span2_end}')) +@then(parsers.parse('the fragment contains span2 with computed times from {frag_span2_begin} to')) def then_fragment_span2_times(test_context, frag_span2_begin, frag_span2_end): if frag_span2_begin == 'deleted': assert_raises(LookupError, test_context['fragment'].get_element_by_id, 'span2') @@ -214,7 +225,13 @@ def then_fragment_span2_times(test_context, frag_span2_begin, frag_span2_end): frag_span2_end).timedelta -@then('the fragment contains span3 with computed times from to ') + +@pytest.fixture +def frag_span3_end(): + return '' + +@then(parsers.parse('the fragment contains span3 with computed times from {frag_span3_begin} to {frag_span3_end}')) +@then(parsers.parse('the fragment contains span3 with computed times from {frag_span3_begin} to')) def then_fragment_span3_times(test_context, frag_span3_begin, frag_span3_end): if frag_span3_begin == 'deleted': assert_raises(LookupError, test_context['fragment'].get_element_by_id, 'span3') @@ -228,7 +245,7 @@ def then_fragment_span3_times(test_context, frag_span3_begin, frag_span3_end): frag_span3_end).timedelta -@then('the fragment only contains styles ') +@then(parsers.parse('the fragment only contains styles {frag_styles}')) def then_fragment_styles_present(test_context, frag_styles): fragment = test_context['fragment'] styling = fragment.binding.head.styling @@ -245,7 +262,12 @@ def then_fragment_styles_present(test_context, frag_styles): assert set(styles_present) == set(styles_required) -@then('the fragment only contains regions ') +@pytest.fixture +def frag_regions(): + return '' + +@then(parsers.parse('the fragment only contains regions {frag_regions}')) +@then(parsers.parse('the fragment only contains regions')) def then_fragment_regions_present(test_context, frag_regions): fragment = test_context['fragment'] layout = fragment.binding.head.layout diff --git a/testing/bdd/test_sequence_attributes_validation.py b/testing/bdd/test_sequence_attributes_validation.py index 8999370de..c8a0cf993 100644 --- a/testing/bdd/test_sequence_attributes_validation.py +++ b/testing/bdd/test_sequence_attributes_validation.py @@ -1,14 +1,24 @@ -from pytest_bdd import when, scenarios - +from pytest_bdd import when, scenarios, parsers +from pytest import fixture scenarios('features/validation/sequence_id_num.feature') -@when('it has sequence identifier ') +@fixture +def seq_id(): + return '' + +@when(parsers.parse('it has sequence identifier {seq_id}')) +@when(parsers.parse('it has sequence identifier')) def when_sequence_id(seq_id, template_dict): template_dict['sequence_id'] = seq_id -@when('it has sequence number ') +@fixture +def seq_n(): + return '' + +@when(parsers.parse('it has sequence number {seq_n}')) +@when(parsers.parse('it has sequence number')) def when_sequence_number(seq_n, template_dict): template_dict['sequence_num'] = seq_n diff --git a/testing/bdd/test_sequence_identical_timing_model.py b/testing/bdd/test_sequence_identical_timing_model.py index c0d15206f..f7a6f187d 100644 --- a/testing/bdd/test_sequence_identical_timing_model.py +++ b/testing/bdd/test_sequence_identical_timing_model.py @@ -10,7 +10,7 @@ # the sequence is created later from the first document and assigning a fixture # after its creation function is not possible. So we store the documents in an # array and create the sequence in the final thens -@given('a test sequence') +@given('a test sequence', target_fixture='doc_list') def doc_list(template_dict): doc_list = [] return doc_list @@ -21,32 +21,62 @@ def when_sequence_number(sequence_number, template_dict): template_dict['sequence_number'] = sequence_number -@when('it has timeBase ') +@pytest.fixture +def time_base1(): + return None + +@when(parsers.parse('it has timeBase {time_base1}')) +@when(parsers.parse('it has timeBase')) def when_time_base1(time_base1, template_dict): template_dict['time_base'] = time_base1 -@when('it has clockMode ') +@pytest.fixture +def clock_mode1(): + return None + +@when(parsers.parse('it has clockMode {clock_mode1}')) +@when(parsers.parse('it has clockMode')) def when_clock_mode1(clock_mode1, template_dict): template_dict['clock_mode'] = clock_mode1 -@when('it has frameRate ') +@pytest.fixture +def frame_rate1(): + return None + +@when(parsers.parse('it has frameRate {frame_rate1}')) +@when(parsers.parse('it has frameRate')) def when_frame_rate1(frame_rate1, template_dict): template_dict['frame_rate'] = frame_rate1 -@when('it has frameRateMultiplier ') +@pytest.fixture +def frame_rate_multiplier1(): + return None + +@when(parsers.parse('it has frameRateMultiplier {frame_rate_multiplier1}')) +@when(parsers.parse('it has frameRateMultiplier')) def when_frame_rate_multiplier1(frame_rate_multiplier1, template_dict): template_dict['frame_rate_multiplier'] = frame_rate_multiplier1 -@when('it has dropMode ') +@pytest.fixture +def drop_mode1(): + return None + +@when(parsers.parse('it has dropMode {drop_mode1}')) +@when(parsers.parse('it has dropMode')) def when_drop_mode1(drop_mode1, template_dict): template_dict['drop_mode'] = drop_mode1 -@when('it has markerMode ') +@pytest.fixture +def marker_mode1(): + return None + +@when(parsers.parse('it has markerMode {marker_mode1}')) +@when(parsers.parse('it has markerMode')) def when_marker_mode1(marker_mode1, template_dict): template_dict['marker_mode'] = marker_mode1 @@ -63,37 +93,72 @@ def when_create_new_document(template_dict): template_dict['sequence_num'] = None -@when('it has timeBase ') +@pytest.fixture +def time_base2(): + return None + +@when(parsers.parse('it has timeBase2 {time_base2}')) +@when(parsers.parse('it has timeBase2')) def when_time_base2(time_base2, template_dict): template_dict['time_base'] = time_base2 -@when('it has clockMode ') +@pytest.fixture +def clock_mode2(): + return None + +@when(parsers.parse('it has clockMode2 {clock_mode2}')) +@when(parsers.parse('it has clockMode2')) def when_clock_mode2(clock_mode2, template_dict): template_dict['clock_mode'] = clock_mode2 -@when('it has frameRate ') +@pytest.fixture +def frame_rate2(): + return None + +@when(parsers.parse('it has frameRate2 {frame_rate2}')) +@when(parsers.parse('it has frameRate2')) def when_frame_rate2(frame_rate2, template_dict): template_dict['frame_rate'] = frame_rate2 -@when('it has frameRateMultiplier ') +@pytest.fixture +def frame_rate_multiplier2(): + return None + +@when(parsers.parse('it has frameRateMultiplier2 {frame_rate_multiplier2}')) +@when(parsers.parse('it has frameRateMultiplier2')) def when_frame_rate_multiplier2(frame_rate_multiplier2, template_dict): template_dict['frame_rate_multiplier'] = frame_rate_multiplier2 -@when('it has dropMode ') +@pytest.fixture +def drop_mode2(): + return None + +@when(parsers.parse('it has dropMode2 {drop_mode2}')) +@when(parsers.parse('it has dropMode2')) def when_drop_mode2(drop_mode2, template_dict): template_dict['drop_mode'] = drop_mode2 -@when('it has markerMode ') +@pytest.fixture +def marker_mode2(): + return None + +@when(parsers.parse('it has markerMode2 {marker_mode2}')) +@when(parsers.parse('it has markerMode2')) def when_marker_mode2(marker_mode2, template_dict): template_dict['marker_mode'] = marker_mode2 -@when('it has sequence number ') +@pytest.fixture +def doc2_seqnum(): + return None + +@when(parsers.parse('it has sequenceNumber2 {doc2_seqnum}')) +@when(parsers.parse('it has sequenceNumber2')) def when_doc2_has_seqnum(doc2_seqnum, template_dict): template_dict['sequence_num'] = doc2_seqnum diff --git a/testing/bdd/test_smpte.py b/testing/bdd/test_smpte.py index 4f4efe238..bd038d772 100644 --- a/testing/bdd/test_smpte.py +++ b/testing/bdd/test_smpte.py @@ -1,28 +1,54 @@ -from pytest_bdd import scenarios, when +from pytest_bdd import scenarios, when, parsers +from pytest import fixture scenarios('features/validation/smpte_constraints.feature') -@when('it has timeBase ') +@fixture +def time_base(): + return None + +@when(parsers.parse('it has timeBase {time_base}')) +@when(parsers.parse('it has timeBase')) def when_time_base(time_base, template_dict): template_dict['time_base'] = time_base -@when('it has frameRate ') +@fixture +def frame_rate(): + return None + +@when(parsers.parse('it has frameRate {frame_rate}')) +@when(parsers.parse('it has frameRate')) def when_frame_rate(frame_rate, template_dict): template_dict['frame_rate'] = frame_rate -@when('it has frameRateMultiplier ') +@fixture +def frame_rate_multiplier(): + return None + +@when(parsers.parse('it has frameRateMultiplier {frame_rate_multiplier}')) +@when(parsers.parse('it has frameRateMultiplier')) def when_frame_rate_multiplier(frame_rate_multiplier, template_dict): template_dict['frame_rate_multiplier'] = frame_rate_multiplier -@when('it has dropMode ') +@fixture +def drop_mode(): + return None + +@when(parsers.parse('it has dropMode {drop_mode}')) +@when(parsers.parse('it has dropMode')) def when_drop_mode(drop_mode, template_dict): template_dict['drop_mode'] = drop_mode -@when('it has markerMode ') +@fixture +def marker_mode(): + return None + +@when(parsers.parse('it has markerMode {marker_mode}')) +@when(parsers.parse('it has markerMode')) def when_marker_mode(marker_mode, template_dict): template_dict['marker_mode'] = marker_mode diff --git a/testing/bdd/test_style.py b/testing/bdd/test_style.py index 9a357dab6..b89d205fa 100644 --- a/testing/bdd/test_style.py +++ b/testing/bdd/test_style.py @@ -1,4 +1,5 @@ -from pytest_bdd import when, scenarios, then +from pytest_bdd import when, scenarios, then, parsers +from pytest import fixture from ebu_tt_live.clocks.media import MediaClock from ebu_tt_live.bindings import ebuttdt from ebu_tt_live.documents.converters import EBUTT3EBUTTDConverter @@ -12,83 +13,133 @@ scenarios('features/styles/lineHeight.feature') -@when('it has a cell resolution of ') +@when(parsers.parse('it has a cell resolution of {cell_resolution}')) def when_cell_resolution(template_dict, cell_resolution): template_dict['cell_resolution'] = cell_resolution -@when('it has extent of ') +@fixture +def extent(): + return None + +@when(parsers.parse('it has extent of {extent}')) +@when(parsers.parse('it has extent of')) def when_extent(template_dict, extent): template_dict['extent'] = extent -@when('it contains style S1 with value ') +@fixture +def S1_value(): + return '' + +@when(parsers.parse('it contains style S1 with {style_attribute} value {S1_value}')) +@when(parsers.parse('it contains style S1 with {style_attribute} value')) def when_s1_attr_value(template_dict, style_attribute, S1_value): template_dict['S1_value'] = S1_value template_dict['style_attribute'] = style_attribute -@when('it contains style S2 with value ') +@fixture +def S2_value(): + return '' + +@when(parsers.parse('it contains style S2 with {style_attribute} value {S2_value}')) +@when(parsers.parse('it contains style S2 with {style_attribute} value')) def when_s2_attr_value(template_dict, style_attribute, S2_value): template_dict['S2_value'] = S2_value template_dict['style_attribute'] = style_attribute -@when('it contains style S3 with value ') +@fixture +def S3_value(): + return '' + +@when(parsers.parse('it contains style S3 with {style_attribute} value {S3_value}')) +@when(parsers.parse('it contains style S3 with {style_attribute} value')) def when_s3_attr_value(template_dict, style_attribute, S3_value): template_dict['S3_value'] = S3_value template_dict['style_attribute'] = style_attribute -@when('it contains style S4 with value ') +@fixture +def S4_value(): + return '' + +@when(parsers.parse('it contains style S4 with {style_attribute} value {S4_value}')) +@when(parsers.parse('it contains style S4 with {style_attribute} value')) def when_s4_attr_value(template_dict, style_attribute, S4_value): template_dict['S4_value'] = S4_value template_dict['style_attribute'] = style_attribute -@when('R1 contains value ') +@fixture +def R1_value(): + return '' + +@when(parsers.parse('R1 contains {style_attribute} value {R1_value}')) +@when(parsers.parse('R1 contains {style_attribute} value')) def when_r1_attr_value(template_dict, style_attribute, R1_value): template_dict['R1_value'] = R1_value template_dict['style_attribute'] = style_attribute -@when('S1 contains value ') +@fixture +def S1_value2(): + return '' + +@when(parsers.parse('S1 contains {style_attribute2} value {S1_value2}')) +@when(parsers.parse('S1 contains {style_attribute2} value')) def when_s1_attr2_value(template_dict, style_attribute2, S1_value2): template_dict['S1_value2'] = S1_value2 template_dict['style_attribute2'] = style_attribute2 -@when('S2 contains value ') +@fixture +def S2_value2(): + return '' + +@when(parsers.parse('S2 contains {style_attribute2} value {S2_value2}')) +@when(parsers.parse('S2 contains {style_attribute2} value')) def when_s2_attr2_value(template_dict, style_attribute2, S2_value2): template_dict['S2_value2'] = S2_value2 template_dict['style_attribute2'] = style_attribute2 -@when('S3 contains value ') +@fixture +def S3_value2(): + return '' + +@when(parsers.parse('S3 contains {style_attribute2} value {S3_value2}')) +@when(parsers.parse('S3 contains {style_attribute2} value')) def when_s3_attr2_value(template_dict, style_attribute2, S3_value2): template_dict['S3_value2'] = S3_value2 template_dict['style_attribute2'] = style_attribute2 -@when('S4 contains value ') +@fixture +def S4_value2(): + return '' + +@when(parsers.parse('S4 contains {style_attribute2} value {S4_value2}')) +@when(parsers.parse('S4 contains {style_attribute2} value')) def when_s4_attr2_value(template_dict, style_attribute2, S4_value2): template_dict['S4_value2'] = S4_value2 template_dict['style_attribute2'] = style_attribute2 -@when('it contains style S5 with value ') +@when(parsers.parse('it contains style S5 with {style_attribute} value {S5_value}')) def when_s5_attr_value(template_dict, style_attribute, S5_value): template_dict['S5_value'] = S5_value template_dict['style_attribute'] = style_attribute -@when('it contains style S6 with value ') +@when(parsers.parse('it contains style S6 with {style_attribute} value {S6_value}')) def when_s6_attr_value(template_dict, style_attribute, S6_value): template_dict['S6_value'] = S6_value template_dict['style_attribute'] = style_attribute -@when('the document is converted to EBUTTD with ') +@when(parsers.parse('the document is converted to EBUTTD with {local_time_mapping}')) def when_document_converted(test_context, local_time_mapping): media_clock = MediaClock() media_clock.adjust_time(timedelta(), ebuttdt.LimitedClockTimingType(local_time_mapping).timedelta) diff --git a/testing/bdd/test_timeBase_attribute_mandatory.py b/testing/bdd/test_timeBase_attribute_mandatory.py index c8cae151f..b1896d356 100644 --- a/testing/bdd/test_timeBase_attribute_mandatory.py +++ b/testing/bdd/test_timeBase_attribute_mandatory.py @@ -1,9 +1,14 @@ -from pytest_bdd import when, scenarios - +from pytest_bdd import when, scenarios, parsers +from pytest import fixture scenarios('features/validation/timeBase_attribute_mandatory.feature') -@when('it has ttp:timeBase attribute ') +@fixture +def time_base(): + return '' + +@when(parsers.parse('it has ttp:timeBase attribute {time_base}')) +@when(parsers.parse('it has ttp:timeBase attribute')) def when_has_time_base(time_base, template_dict): template_dict['time_base'] = time_base diff --git a/testing/bdd/test_timeBase_clock_clockMode_mandatory.py b/testing/bdd/test_timeBase_clock_clockMode_mandatory.py index ced7e50d3..ca1257a87 100644 --- a/testing/bdd/test_timeBase_clock_clockMode_mandatory.py +++ b/testing/bdd/test_timeBase_clock_clockMode_mandatory.py @@ -1,14 +1,20 @@ -from pytest_bdd import when, scenarios +from pytest_bdd import when, scenarios, parsers +from pytest import fixture scenarios('features/validation/timeBase_clock_clockMode_mandatory.feature') -@when('it has ttp:timeBase attribute ') +@when(parsers.parse('it has ttp:timeBase attribute {time_base}')) def when_has_time_base(time_base, template_dict): template_dict['time_base'] = time_base -@when('it has ttp:clockMode attribute ') +@fixture +def clock_mode(): + return '' + +@when(parsers.parse('it has ttp:clockMode attribute {clock_mode}')) +@when(parsers.parse('it has ttp:clockMode attribute')) def when_has_clock_mode(clock_mode, template_dict): template_dict['clock_mode'] = clock_mode diff --git a/testing/bdd/test_timeBase_timeformat.py b/testing/bdd/test_timeBase_timeformat.py index d8f72bf3c..212c2b4bc 100644 --- a/testing/bdd/test_timeBase_timeformat.py +++ b/testing/bdd/test_timeBase_timeformat.py @@ -1,53 +1,100 @@ -from pytest_bdd import scenarios, when +from pytest_bdd import scenarios, when, parsers +from pytest import fixture + scenarios('features/validation/timeBase_timeformat_constraints.feature') -@when('it has timeBase ') +@when(parsers.parse('it has timeBase {time_base}')) def when_time_base(time_base, template_dict): template_dict['time_base'] = time_base -@when('it has body begin time ') +@fixture +def body_begin(): + return '' + +@when(parsers.parse('it has body begin time {body_begin}')) +@when(parsers.parse('it has body begin time')) def when_body_begin(body_begin, template_dict): template_dict['body_begin'] = body_begin -@when('it has body end time ') +@fixture +def body_end(): + return '' + +@when(parsers.parse('it has body end time {body_end}')) +@when(parsers.parse('it has body end time')) def when_body_end(body_end, template_dict): template_dict['body_end'] = body_end -@when('it has body duration ') +@fixture +def body_dur(): + return '' + +@when(parsers.parse('it has body duration {body_dur}')) +@when(parsers.parse('it has body duration')) def when_body_dur(body_dur, template_dict): template_dict['body_dur'] = body_dur -@when('it has div begin time ') +@fixture +def div_begin(): + return '' + +@when(parsers.parse('it has div begin time {div_begin}')) +@when(parsers.parse('it has div begin time')) def when_div_begin(div_begin, template_dict): template_dict['div_begin'] = div_begin -@when('it has div end time ') +@fixture +def div_end(): + return '' + +@when(parsers.parse('it has div end time {div_end}')) +@when(parsers.parse('it has div end time')) def when_div_end(div_end, template_dict): template_dict['div_end'] = div_end -@when('it has p begin time ') +@fixture +def p_begin(): + return '' + +@when(parsers.parse('it has p begin time {p_begin}')) +@when(parsers.parse('it has p begin time')) def when_p_begin(p_begin, template_dict): template_dict['p_begin'] = p_begin -@when('it has p end time ') +@fixture +def p_end(): + return '' + +@when(parsers.parse('it has p end time {p_end}')) +@when(parsers.parse('it has p end time')) def when_p_end(p_end, template_dict): template_dict['p_end'] = p_end -@when('it has span begin time ') +@fixture +def span_begin(): + return '' + +@when(parsers.parse('it has span begin time {span_begin}')) +@when(parsers.parse('it has span begin time')) def when_span_begin(span_begin, template_dict): template_dict['span_begin'] = span_begin -@when('it has span end time ') +@fixture +def span_end(): + return '' + +@when(parsers.parse('it has span end time {span_end}')) +@when(parsers.parse('it has span end time')) def when_span_end(span_end, template_dict): template_dict['span_end'] = span_end diff --git a/testing/bdd/test_time_regex_parsing.py b/testing/bdd/test_time_regex_parsing.py index bf94dfaf5..784e5e89a 100644 --- a/testing/bdd/test_time_regex_parsing.py +++ b/testing/bdd/test_time_regex_parsing.py @@ -14,12 +14,12 @@ ] -@when('it has timeBase ') +@when(parsers.parse('it has timeBase {time_base}')) def when_time_base(time_base, template_dict): template_dict['time_base'] = time_base -@when('it has body begin time ') +@when(parsers.parse('it has body begin time {body_begin}')) def when_body_begin(body_begin, template_dict): template_dict['body_begin'] = body_begin diff --git a/testing/bdd/test_websocket_carriage_config.py b/testing/bdd/test_websocket_carriage_config.py index d9af961ff..36611aaa2 100644 --- a/testing/bdd/test_websocket_carriage_config.py +++ b/testing/bdd/test_websocket_carriage_config.py @@ -1,6 +1,6 @@ import os from jinja2 import Environment, FileSystemLoader -from pytest_bdd import scenarios, given, when, then +from pytest_bdd import scenarios, given, when, then, parsers from pytest import fixture import socket @@ -13,7 +13,7 @@ def config_dict(): return dict() -@given('a configuration file ') +@given(parsers.parse('a configuration file {config_file}'), target_fixture='given_config_file') def given_config_file(config_file): cur_dir = os.path.dirname(os.path.abspath(__file__)) j2_env = Environment(loader=FileSystemLoader(os.path.join(cur_dir, 'templates')), @@ -39,17 +39,17 @@ def when_producer_listens_port(): pass -@when('the consumer connects to the port with ') +@when(parsers.parse('the consumer connects to the port with {client_url_path}')) def when_consumer_connects_port(client_url_path): pass -@when('producer sends document with ') +@when(parsers.parse('producer sends document with {sequence_number_1}')) def when_producer_sends_document1(sequence_number_1): pass -@when('producer sends document with ') +@when(parsers.parse('producer sends document with {sequence_number_2}')) def when_producer_sends_document2(sequence_number_2): pass diff --git a/testing/bdd/test_xml_lang_attribute.py b/testing/bdd/test_xml_lang_attribute.py index ac94b2d72..a15de5d60 100644 --- a/testing/bdd/test_xml_lang_attribute.py +++ b/testing/bdd/test_xml_lang_attribute.py @@ -1,9 +1,15 @@ -from pytest_bdd import when, scenarios +from pytest_bdd import when, scenarios, parsers +from pytest import fixture scenarios('features/validation/xml_lang_attribute.feature') -@when('it has xml:lang attribute ') +@fixture +def lang(): + return None + +@when(parsers.parse('it has xml:lang attribute {lang}')) +@when(parsers.parse('it has xml:lang attribute')) def when_lang(lang, template_dict): template_dict['lang'] = lang From 5011a16b3dc790c571234b92391da9712c70a064 Mon Sep 17 00:00:00 2001 From: Nigel Megitt Date: Tue, 7 Dec 2021 12:23:32 +0000 Subject: [PATCH 7/7] Delete commented line --- ebu_tt_live/documents/ebutt3_segmentation.py | 1 - 1 file changed, 1 deletion(-) diff --git a/ebu_tt_live/documents/ebutt3_segmentation.py b/ebu_tt_live/documents/ebutt3_segmentation.py index b9e534283..509177bc6 100644 --- a/ebu_tt_live/documents/ebutt3_segmentation.py +++ b/ebu_tt_live/documents/ebutt3_segmentation.py @@ -157,7 +157,6 @@ def _prune_orphan_elements(self): if len(self._segment.head.styling.style) == 0: self._segment.head.styling = None if len(self._segment.head.layout.region) == 0: - self._segment.head.layout.region = None self._segment.head.layout = None def compute_document_segment(self):