/* * Copyright (C) 2015-present ScyllaDB * * Modified by ScyllaDB */ /* * SPDX-License-Identifier: (LicenseRef-ScyllaDB-Source-Available-1.0 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 "service/qos/service_level_controller.hh" #include "types/types.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, std::optional service_level, std::optional&& concurrency) : _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)} , _service_level(std::move(service_level)) , _concurrency{std::move(concurrency)} { } 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); } bool attributes::is_service_level_set() const { return bool(_service_level); } bool attributes::is_concurrency_set() const { return bool(_concurrency); } 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)); } qos::service_level_options attributes::get_service_level(qos::service_level_controller& sl_controller) const { auto sl_name = *_service_level; if (!sl_controller.has_service_level(sl_name)) { throw exceptions::invalid_request_exception(format("Service level {} doesn't exist", sl_name)); } return sl_controller.get_service_level(sl_name).slo; } std::optional attributes::get_concurrency(const query_options& options) const { if (!_concurrency.has_value()) { return std::nullopt; } cql3::raw_value concurrency_raw = expr::evaluate(*_concurrency, options); if (concurrency_raw.is_null()) { throw exceptions::invalid_request_exception("Invalid null value of concurrency"); } int32_t concurrency; try { concurrency = concurrency_raw.view().validate_and_deserialize(*int32_type); } catch (marshal_exception& e) { throw exceptions::invalid_request_exception("Invalid concurrency value"); } if (concurrency <= 0) { throw exceptions::invalid_request_exception("Concurrency must be a positive integer"); } return concurrency; } 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); } if (_concurrency.has_value()) { expr::fill_prepare_context(*_concurrency, 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, conc; 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"); } if (concurrency.has_value()) { conc = prepare_expression(*concurrency, db, ks_name, nullptr, concurrency_receiver(ks_name, cf_name)); verify_no_aggregate_functions(*concurrency, "USING clause"); } return std::unique_ptr{new attributes{std::move(ts), std::move(ttl), std::move(to), std::move(service_level), std::move(conc)}}; } 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); } lw_shared_ptr attributes::raw::concurrency_receiver(const sstring& ks_name, const sstring& cf_name) const { return make_lw_shared(ks_name, cf_name, ::make_shared("[concurrency]", true), data_type_for()); } }