1313# limitations under the License.
1414
1515import os
16+ from distutils .log import fatal
17+
1618from cassandra .cluster import Cluster
1719
1820from tests import connection_class , EVENT_LOOP_MANAGER
@@ -299,6 +301,8 @@ def get_unsupported_lower_protocol():
299301 This is used to determine the lowest protocol version that is NOT
300302 supported by the version of C* running
301303 """
304+ if SCYLLA_VERSION is not None :
305+ return 2
302306 if CASSANDRA_VERSION >= Version ('3.0' ):
303307 return 2
304308 else :
@@ -310,7 +314,8 @@ def get_unsupported_upper_protocol():
310314 This is used to determine the highest protocol version that is NOT
311315 supported by the version of C* running
312316 """
313-
317+ if SCYLLA_VERSION is not None :
318+ return 5
314319 if CASSANDRA_VERSION >= Version ('4.0-a' ):
315320 if DSE_VERSION :
316321 return None
@@ -389,6 +394,64 @@ def _id_and_mark(f):
389394requires_custom_payload = pytest .mark .skipif (SCYLLA_VERSION is not None or PROTOCOL_VERSION < 4 ,
390395 reason = 'Scylla does not support custom payloads. Cassandra requires native protocol v4.0+' )
391396xfail_scylla = lambda reason , * args , ** kwargs : pytest .mark .xfail (SCYLLA_VERSION is not None , reason = reason , * args , ** kwargs )
397+
398+
399+ def xfail_scylla_version_lt (reason , oss_scylla_version , ent_scylla_version , * args , ** kwargs ):
400+ """
401+ It is used to mark tests that are going to fail on certain scylla versions.
402+
403+ :param reason: message to fail test with
404+ :param oss_scylla_version: str, oss version from which test supposed to succeed
405+ :param ent_scylla_version: str|list<str>|tuple<str>, enterprise versions from which test supposed to succeed.
406+ Should contain at least first major: `x.1.1` or `x.1` or `x`.
407+ If current version is lower than `first major` it expects it to fail
408+ For rest versions it looks for a version with same `major` and `minor` if records exists and `micro` is less,
409+ it expects it to fail, otherwise expects it to succeed.
410+ """
411+ if not reason .startswith ("scylladb/scylladb#" ):
412+ raise ValueError ('reason should start with scylladb/scylladb#<issue-id> to reference issue in scylla repo' )
413+
414+ if not isinstance (ent_scylla_version , (tuple , list )):
415+ ent_scylla_version = (ent_scylla_version ,)
416+ first_major = None
417+ ent_scylla_version = [Version (v ) for v in ent_scylla_version ]
418+
419+ # Enterprise releases are tricky, here are the rules:
420+ # 1. If something is fixed in `a.b.c` it is also fixed in `a.b.d` where `c <= d`
421+ # 2. If something is fixed in `a.1.1` it is also fixed in `c.d.f` where `Version(a.1.1)` <= `Version(c.d.f`)`
422+ # To make version matching work properly of enterprise we need to have a first major version `x.1.1` where it was fixed.
423+ for ent_v in ent_scylla_version :
424+ if ent_v < Version ("2018.1" ):
425+ raise ValueError ("ent_scylla_version should contain only enterprise versions, example: 2024.1.1" )
426+ if ent_v .minor in (1 ,0 ) and ent_v .micro in (1 ,0 ):
427+ first_major = Version (str (ent_v .major )+ "0.0" )
428+
429+ if not first_major :
430+ raise ValueError ("ent_scylla_version should contain first major LTS: `x.1.1`" )
431+
432+ if SCYLLA_VERSION is None :
433+ return pytest .mark .skipif (False , reason = "It is just a NoOP Decor, should not skip anything" )
434+
435+ current_version = Version (get_scylla_version (SCYLLA_VERSION ) if SCYLLA_VERSION is not None else '0.0.0' )
436+ is_enterprise = current_version > Version ("2018.1" )
437+ if not is_enterprise :
438+ return pytest .mark .xfail (current_version < Version (oss_scylla_version ),
439+ reason = reason , * args , ** kwargs )
440+
441+ if first_major < current_version :
442+ # Expect test to fail if it is less then first major
443+ return pytest .mark .xfail (False , reason = reason , * args , ** kwargs )
444+
445+ for ent_v in ent_scylla_version :
446+ # Expect test to fail if there is any record with same major and minor and micro is higher
447+ if ent_v .major == current_version .major and ent_v .minor == current_version .minor :
448+ return pytest .mark .xfail (current_version < ent_v , reason = reason , * args , ** kwargs )
449+
450+ # Noop
451+ # Expect test to complete successfully
452+ return pytest .mark .xfail (True , reason = reason , * args , ** kwargs )
453+
454+
392455incorrect_test = lambda reason = 'This test seems to be incorrect and should be fixed' , * args , ** kwargs : pytest .mark .xfail (reason = reason , * args , ** kwargs )
393456
394457pypy = unittest .skipUnless (platform .python_implementation () == "PyPy" , "Test is skipped unless it's on PyPy" )
0 commit comments