diff --git a/doc/modules/cassandra/examples/CQL/to_date.cql b/doc/modules/cassandra/examples/CQL/to_date.cql new file mode 100644 index 000000000000..160dcaab6726 --- /dev/null +++ b/doc/modules/cassandra/examples/CQL/to_date.cql @@ -0,0 +1 @@ +SELECT id, to_date(create_ts) FROM myTable diff --git a/doc/modules/cassandra/pages/developing/cql/functions.adoc b/doc/modules/cassandra/pages/developing/cql/functions.adoc index 75786de271a3..d74beb1446fc 100644 --- a/doc/modules/cassandra/pages/developing/cql/functions.adoc +++ b/doc/modules/cassandra/pages/developing/cql/functions.adoc @@ -184,12 +184,12 @@ time where the function is invoked: |=== |Function name |Output type -| `current_timestamp` | `timestamp` - | `current_date` | `date` | `current_time` | `time` +| `current_timestamp` | `timestamp` + | `current_timeuuid` | `timeUUID` |=== @@ -223,6 +223,13 @@ A number of functions are provided to convert a `timeuuid`, a `timestamp` or a ` | `to_unix_timestamp` | `date` | Converts the `date` argument into a `bigInt` raw value |=== +For example, a timestamp can be converted to a date with the following: + +[source,cql] +---- +include::cassandra:example$CQL/to_date.cql[] +---- + ==== Blob conversion functions A number of functions are provided to convert the native types into diff --git a/pylib/cqlshlib/cql3handling.py b/pylib/cqlshlib/cql3handling.py index 457582f3d379..101c39ca6cde 100644 --- a/pylib/cqlshlib/cql3handling.py +++ b/pylib/cqlshlib/cql3handling.py @@ -602,7 +602,7 @@ def cf_prop_val_mapender_completer(ctxt, cass): @completer_for('tokenDefinition', 'token') def token_word_completer(ctxt, cass): - return ['token('] + return ['TOKEN'] @completer_for('simpleStorageType', 'typename') @@ -741,12 +741,13 @@ def working_on_keyspace(ctxt): ; ::= ( "AND" )* ; - ::= [rel_lhs]= ( "[" "]" )? ( "=" | "<" | ">" | "<=" | ">=" | "!=" | ( "NOT" )? "CONTAINS" ( "KEY" )? ) + ::= [rel_lhs]= ( "[" "]" )? ( "=" | "<" | ">" | "<=" | ">=" | "!=" | ( "NOT" )? "CONTAINS" ( "KEY" )? ) ( | ) | token="TOKEN" "(" [rel_tokname]= ( "," [rel_tokname]= )* ")" ("=" | "<" | ">" | "<=" | ">=") | [rel_lhs]= (( "NOT" )? "IN" ) "(" ( "," )* ")" | [rel_lhs]= "BETWEEN" "AND" + | ; ::= "DISTINCT"? ("AS" )? ("," ("AS" )?)* | "*" @@ -755,14 +756,20 @@ def working_on_keyspace(ctxt): ; ::= [colname]= ( "[" ( ( ".." "]" )? | ".." ) )? | - | "WRITETIME" "(" [colname]= ")" - | "MAXWRITETIME" "(" [colname]= ")" - | "TTL" "(" [colname]= ")" - | "COUNT" "(" star=( "*" | "1" ) ")" | "CAST" "(" "AS" ")" + | "TTL" "(" [colname]= ")" + | "TOKEN" "(" [colname]= ")" + | + | + | + | + | + | + | | | ; + ::= "(" ( ( "," )* )? ")" ; ::= [ordercol]= ( "ASC" | "DESC" )? @@ -775,6 +782,60 @@ def working_on_keyspace(ctxt): ::= [groupcol]= | ; + + ::= "COUNT" "(" star=( "*" | "1" ) ")" + | "AVG" "(" [colname]= ")" + | "MIN" "(" [colname]= ")" + | "MAX" "(" [colname]= ")" + | "SUM" "(" [colname]= ")" + ; + + ::= "ABS" "(" [colname]= ")" + | "EXP" "(" [colname]= ")" + | "LOG" "(" [colname]= ")" + | "LOG10" "(" [colname]= ")" + | "ROUND" "(" [colname]= ")" + ; + + ::= "MAP_KEYS" "(" [colname]= ")" + | "MAP_VALUES" "(" [colname]= ")" + | "COLLECTION_AVG" "(" [colname]= ")" + | "COLLECTION_COUNT" "(" [colname]= ")" + | "COLLECTION_MIN" "(" [colname]= ")" + | "COLLECTION_MAX" "(" [colname]= ")" + | "COLLECTION_SUM" "(" [colname]= ")" + ; + + ::= "CURRENT_DATE()" + | "CURRENT_TIME()" + | "CURRENT_TIMESTAMP()" + | "CURRENT_TIMEUUID()" + ; + + ::= "MASK_DEFAULT" "(" [colname]= ")" + | "MASK_HASH" "(" [colname]= ")" + | "MASK_INNER" "(" [colname]= "," "," ")" + | "MASK_NULL" "(" [colname]= ")" + | "MASK_REPLACE" "(" [colname]= "," ")" + | "MASK_OUTER" "(" [colname]= "," "," ")" + ; + + ::= "TO_DATE" "(" [colname]= ")" + | "TO_TIMESTAMP" "(" [colname]= ")" + | "TO_UNIX_TIMESTAMP" "(" [colname]= ")" + ; + + ::= "MAX_TIMEUUID" "(" [colname]= ")" + | "MIN_TIMEUUID" "(" [colname]= ")" + ; + + ::= "MAX_WRITETIME" "(" [colname]= ")" + | "MIN_WRITETIME" "(" [colname]= ")" + | "WRITETIME" "(" [colname]= ")" + ; + ::= | + ; + ''' @@ -867,7 +928,7 @@ def select_group_column_completer(ctxt, cass): @completer_for('relation', 'token') def relation_token_word_completer(ctxt, cass): - return ['TOKEN('] + return ['TOKEN'] @completer_for('relation', 'rel_tokname') @@ -1001,7 +1062,7 @@ def insert_option_completer(ctxt, cass): @completer_for('updateStatement', 'updateopt') def update_option_completer(ctxt, cass): - opts = set('TIMESTAMP TTL'.split()) + opts = {'TIMESTAMP', 'TTL'} for opt in ctxt.get_binding('updateopt', ()): opts.discard(opt.split()[0]) return opts diff --git a/pylib/cqlshlib/cqlshmain.py b/pylib/cqlshlib/cqlshmain.py index 2cac58ef22c4..2bc79f3c377b 100755 --- a/pylib/cqlshlib/cqlshmain.py +++ b/pylib/cqlshlib/cqlshmain.py @@ -383,7 +383,7 @@ def check_build_versions(self): baseversion = baseversion[0:extra] if baseversion != build_version: print("WARNING: cqlsh was built against {}, but this server is {}. All features may not work!" - .format(build_version, baseversion)) # ToDo: use file=sys.stderr) + .format(build_version, baseversion), file=sys.stderr) @property def batch_mode(self): diff --git a/pylib/cqlshlib/test/test_cqlsh_completion.py b/pylib/cqlshlib/test/test_cqlsh_completion.py index 112474e7c712..53ab1908ed45 100644 --- a/pylib/cqlshlib/test/test_cqlsh_completion.py +++ b/pylib/cqlshlib/test/test_cqlsh_completion.py @@ -114,7 +114,8 @@ def _get_completions(self, inputstring, split_completed_lines=True): def _trycompletions_inner(self, inputstring, immediate='', choices=(), other_choices_ok=False, - split_completed_lines=True): + split_completed_lines=True, + ignore_system_keyspaces=False): """ Test tab completion in cqlsh. Enters in the text in inputstring, then simulates a tab keypress to see what is immediately completed (this @@ -132,17 +133,22 @@ def _trycompletions_inner(self, inputstring, immediate='', choices=(), self.assertEqual(completed, immediate, msg=msg) return + if ignore_system_keyspaces: + completed = list(filter(lambda s: not s.startswith('system'), completed)) + if other_choices_ok: self.assertEqual(set(choices), completed.intersection(choices)) else: self.assertEqual(set(choices), set(completed)) def trycompletions(self, inputstring, immediate='', choices=(), - other_choices_ok=False, split_completed_lines=True): + other_choices_ok=False, split_completed_lines=True, + ignore_system_keyspaces=False): try: self._trycompletions_inner(inputstring, immediate, choices, other_choices_ok=other_choices_ok, - split_completed_lines=split_completed_lines) + split_completed_lines=split_completed_lines, + ignore_system_keyspaces=ignore_system_keyspaces) finally: try: self.cqlsh.send(CTRL_C) # cancel any current line @@ -175,7 +181,43 @@ def test_complete_in_uuid(self): pass def test_complete_in_select(self): - pass + self.trycompletions('SELECT ', + choices=('*', '', + '-', '', '', '', '', + '', '', '', + 'ABS', 'AVG', 'CAST', 'COUNT', 'DISTINCT', + 'EXP', 'JSON', 'LOG', 'LOG10', + 'MAP_KEYS', 'MAP_VALUES', + 'MIN', 'MAX', + 'MIN_WRITETIME', 'MAX_WRITETIME', + 'ROUND', 'SUM', 'TOKEN', + 'TO_DATE', 'TO_TIMESTAMP', 'TO_UNIX_TIMESTAMP', + 'TTL', 'WRITETIME', + 'COLLECTION_AVG', 'COLLECTION_COUNT', 'COLLECTION_MAX', + 'COLLECTION_MIN', 'COLLECTION_SUM', + 'MASK_DEFAULT', 'MASK_HASH', 'MASK_INNER', 'MASK_NULL', + 'MASK_OUTER', 'MASK_REPLACE', + '[', '{', 'false', 'true', 'NULL' + ), + other_choices_ok=True + ) + + def test_complete_in_select_where(self): + self.trycompletions('SELECT * FROM system.peers WHERE ', + choices=('', '', 'peer', 'CURRENT_DATE()', 'CURRENT_TIME()', + 'CURRENT_TIMEUUID()', 'CURRENT_TIMESTAMP()', 'TOKEN', + 'MIN_TIMEUUID', 'MAX_TIMEUUID') + ) + + def test_complete_in_select_where_equal(self): + self.trycompletions('SELECT * FROM system.peers WHERE rack = ', + choices=('-', '', '', '', '', + '', '', '', + '[', '{', 'false', 'true', 'NULL', + 'TOKEN', 'MIN_TIMEUUID', 'MAX_TIMEUUID', + 'CURRENT_DATE()', 'CURRENT_TIME()', 'CURRENT_TIMEUUID()', 'CURRENT_TIMESTAMP()' + ) + ) def test_complete_in_insert(self): self.trycompletions('INSERT INTO ', @@ -376,7 +418,8 @@ def test_complete_in_update(self): self.trycompletions("UPDATE empty_table SET lonelycol = 'eggs'", choices=[',', 'WHERE']) self.trycompletions("UPDATE empty_table SET lonelycol = 'eggs' WHERE ", - choices=['TOKEN(', 'lonelykey']) + choices=['CURRENT_DATE()', 'CURRENT_TIME()', 'CURRENT_TIMESTAMP()', + 'CURRENT_TIMEUUID()', 'TOKEN', 'MIN_TIMEUUID', 'MAX_TIMEUUID', 'lonelykey']) self.trycompletions("UPDATE empty_table SET lonelycol = 'eggs' WHERE lonel", immediate='ykey ') @@ -385,7 +428,8 @@ def test_complete_in_update(self): self.trycompletions("UPDATE empty_table SET lonelycol = 'eggs' WHERE lonelykey = 0.0 ", choices=['AND', 'IF', ';']) self.trycompletions("UPDATE empty_table SET lonelycol = 'eggs' WHERE lonelykey = 0.0 AND ", - choices=['TOKEN(', 'lonelykey']) + choices=['CURRENT_DATE()', 'CURRENT_TIME()', 'CURRENT_TIMESTAMP()', + 'CURRENT_TIMEUUID()', 'TOKEN', 'MIN_TIMEUUID', 'MAX_TIMEUUID', 'lonelykey']) self.trycompletions("UPDATE empty_table SET lonelycol = 'eggs' WHERE TOKEN(lonelykey ", choices=[',', ')']) @@ -397,7 +441,7 @@ def test_complete_in_update(self): choices=['EXISTS', '', '']) self.trycompletions("UPDATE empty_table SET lonelycol = 'eggs' WHERE TOKEN(lonelykey) <= TOKEN(13) IF EXISTS ", - choices=['>=', '!=', '<=', 'IN','[', ';', '=', '<', '>', '.', 'CONTAINS']) + choices=['>=', '!=', '<=', 'IN', '[', ';', '=', '<', '>', '.', 'CONTAINS']) self.trycompletions("UPDATE empty_table SET lonelycol = 'eggs' WHERE TOKEN(lonelykey) <= TOKEN(13) IF lonelykey ", choices=['>=', '!=', '<=', 'IN', '=', '<', '>', 'CONTAINS']) @@ -461,10 +505,11 @@ def test_complete_in_delete(self): self.trycompletions('DELETE FROM twenty_rows_composite_table USING TIMESTAMP 0 ', immediate='WHERE ') self.trycompletions('DELETE FROM twenty_rows_composite_table USING TIMESTAMP 0 WHERE ', - choices=['a', 'b', 'TOKEN(']) + choices=['a', 'b', 'CURRENT_DATE()', 'CURRENT_TIME()', 'CURRENT_TIMESTAMP()', + 'CURRENT_TIMEUUID()', 'MAX_TIMEUUID', 'MIN_TIMEUUID', 'TOKEN']) self.trycompletions('DELETE FROM twenty_rows_composite_table USING TIMESTAMP 0 WHERE a ', - choices=['<=', '>=', 'BETWEEN', 'CONTAINS', 'IN', 'NOT' , '[', '=', '<', '>', '!=']) + choices=['<=', '>=', 'BETWEEN', 'CONTAINS', 'IN', 'NOT', '[', '=', '<', '>', '!=']) self.trycompletions('DELETE FROM twenty_rows_composite_table USING TIMESTAMP 0 WHERE TOKEN(', immediate='a ') @@ -476,7 +521,7 @@ def test_complete_in_delete(self): choices=['>=', '<=', '=', '<', '>']) self.trycompletions('DELETE FROM twenty_rows_composite_table USING TIMESTAMP 0 WHERE TOKEN(a) >= ', choices=['false', 'true', '', - 'token(', '-', '', 'TOKEN', + '-', '', 'TOKEN', '', '', '{', '[', 'NULL', '', '', ''])