/* * Copyright (C) 2015-present ScyllaDB * * Modified by ScyllaDB */ /* * SPDX-License-Identifier: (AGPL-3.0-or-later and Apache-2.0) */ #include "cql3/attributes.hh" #include "cql3/column_identifier.hh" #include "cql3/expr/evaluate.hh" #include "cql3/expr/expr-utils.hh" #include "exceptions/exceptions.hh" #include namespace cql3 { std::unique_ptr attributes::none() { return std::unique_ptr{new attributes{{}, {}, {}}}; } attributes::attributes(std::optional&& timestamp, std::optional&& time_to_live, std::optional&& timeout) : _timestamp_unset_guard(timestamp) , _timestamp{std::move(timestamp)} , _time_to_live_unset_guard(time_to_live) , _time_to_live{std::move(time_to_live)} , _timeout{std::move(timeout)} { } bool attributes::is_timestamp_set() const { return bool(_timestamp); } bool attributes::is_time_to_live_set() const { return bool(_time_to_live); } bool attributes::is_timeout_set() const { return bool(_timeout); } int64_t attributes::get_timestamp(int64_t now, const query_options& options) { if (!_timestamp.has_value() || _timestamp_unset_guard.is_unset(options)) { return now; } cql3::raw_value tval = expr::evaluate(*_timestamp, options); if (tval.is_null()) { throw exceptions::invalid_request_exception("Invalid null value of timestamp"); } try { return tval.view().validate_and_deserialize(*long_type); } catch (marshal_exception& e) { throw exceptions::invalid_request_exception("Invalid timestamp value"); } } std::optional attributes::get_time_to_live(const query_options& options) { if (!_time_to_live.has_value() || _time_to_live_unset_guard.is_unset(options)) return std::nullopt; cql3::raw_value tval = expr::evaluate(*_time_to_live, options); if (tval.is_null()) { throw exceptions::invalid_request_exception("Invalid null value of TTL"); } int32_t ttl; try { ttl = tval.view().validate_and_deserialize(*int32_type); } catch (marshal_exception& e) { throw exceptions::invalid_request_exception("Invalid TTL value"); } if (ttl < 0) { throw exceptions::invalid_request_exception("A TTL must be greater or equal to 0"); } if (ttl > max_ttl.count()) { throw exceptions::invalid_request_exception("ttl is too large. requested (" + std::to_string(ttl) + ") maximum (" + std::to_string(max_ttl.count()) + ")"); } return ttl; } db::timeout_clock::duration attributes::get_timeout(const query_options& options) const { cql3::raw_value timeout = expr::evaluate(*_timeout, options); if (timeout.is_null()) { throw exceptions::invalid_request_exception("Timeout value cannot be null"); } cql_duration duration = timeout.view().deserialize(*duration_type); if (duration.months || duration.days) { throw exceptions::invalid_request_exception("Timeout values cannot be expressed in days/months"); } if (duration.nanoseconds % 1'000'000 != 0) { throw exceptions::invalid_request_exception("Timeout values cannot have granularity finer than milliseconds"); } if (duration.nanoseconds < 0) { throw exceptions::invalid_request_exception("Timeout values must be non-negative"); } return std::chrono::duration_cast(std::chrono::nanoseconds(duration.nanoseconds)); } void attributes::fill_prepare_context(prepare_context& ctx) { if (_timestamp.has_value()) { expr::fill_prepare_context(*_timestamp, ctx); } if (_time_to_live.has_value()) { expr::fill_prepare_context(*_time_to_live, ctx); } if (_timeout.has_value()) { expr::fill_prepare_context(*_timeout, ctx); } } std::unique_ptr attributes::raw::prepare(data_dictionary::database db, const sstring& ks_name, const sstring& cf_name) const { std::optional ts, ttl, to; if (timestamp.has_value()) { ts = prepare_expression(*timestamp, db, ks_name, nullptr, timestamp_receiver(ks_name, cf_name)); verify_no_aggregate_functions(*ts, "USING clause"); } if (time_to_live.has_value()) { ttl = prepare_expression(*time_to_live, db, ks_name, nullptr, time_to_live_receiver(ks_name, cf_name)); verify_no_aggregate_functions(*time_to_live, "USING clause"); } if (timeout.has_value()) { to = prepare_expression(*timeout, db, ks_name, nullptr, timeout_receiver(ks_name, cf_name)); verify_no_aggregate_functions(*timeout, "USING clause"); } return std::unique_ptr{new attributes{std::move(ts), std::move(ttl), std::move(to)}}; } lw_shared_ptr attributes::raw::timestamp_receiver(const sstring& ks_name, const sstring& cf_name) const { return make_lw_shared(ks_name, cf_name, ::make_shared("[timestamp]", true), data_type_for()); } lw_shared_ptr attributes::raw::time_to_live_receiver(const sstring& ks_name, const sstring& cf_name) const { return make_lw_shared(ks_name, cf_name, ::make_shared("[ttl]", true), data_type_for()); } lw_shared_ptr attributes::raw::timeout_receiver(const sstring& ks_name, const sstring& cf_name) const { return make_lw_shared(ks_name, cf_name, ::make_shared("[timeout]", true), duration_type); } }