/* * Copyright (C) 2014 Cloudius Systems, Ltd. */ #include "mutation.hh" mutation::mutation(dht::decorated_key key, schema_ptr schema) : _schema(std::move(schema)) , _dk(std::move(key)) , _p(_schema) { } mutation::mutation(partition_key key_, schema_ptr schema) : _schema(std::move(schema)) , _dk(dht::global_partitioner().decorate_key(*_schema, std::move(key_))) , _p(_schema) { } mutation::mutation(schema_ptr schema, dht::decorated_key key, mutation_partition mp) : _schema(std::move(schema)), _dk(std::move(key)), _p(std::move(mp)) { } void mutation::set_static_cell(const column_definition& def, atomic_cell_or_collection value) { _p.static_row().apply(def, std::move(value)); } void mutation::set_static_cell(const bytes& name, const boost::any& value, api::timestamp_type timestamp, ttl_opt ttl) { auto column_def = _schema->get_column_definition(name); if (!column_def) { throw std::runtime_error(sprint("no column definition found for '%s'", name)); } if (!column_def->is_static()) { throw std::runtime_error(sprint("column '%s' is not static", name)); } _p.static_row().apply(*column_def, atomic_cell::make_live(timestamp, column_def->type->decompose(value), ttl)); } void mutation::set_clustered_cell(const exploded_clustering_prefix& prefix, const column_definition& def, atomic_cell_or_collection value) { auto& row = _p.clustered_row(clustering_key::from_clustering_prefix(*_schema, prefix)).cells(); row.apply(def, std::move(value)); } void mutation::set_clustered_cell(const clustering_key& key, const bytes& name, const boost::any& value, api::timestamp_type timestamp, ttl_opt ttl) { auto column_def = _schema->get_column_definition(name); if (!column_def) { throw std::runtime_error(sprint("no column definition found for '%s'", name)); } return set_clustered_cell(key, *column_def, atomic_cell::make_live(timestamp, column_def->type->decompose(value), ttl)); } void mutation::set_clustered_cell(const clustering_key& key, const column_definition& def, atomic_cell_or_collection value) { auto& row = _p.clustered_row(key).cells(); row.apply(def, std::move(value)); } void mutation::set_cell(const exploded_clustering_prefix& prefix, const bytes& name, const boost::any& value, api::timestamp_type timestamp, ttl_opt ttl) { auto column_def = _schema->get_column_definition(name); if (!column_def) { throw std::runtime_error(sprint("no column definition found for '%s'", name)); } return set_cell(prefix, *column_def, atomic_cell::make_live(timestamp, column_def->type->decompose(value), ttl)); } void mutation::set_cell(const exploded_clustering_prefix& prefix, const column_definition& def, atomic_cell_or_collection value) { if (def.is_static()) { set_static_cell(def, std::move(value)); } else if (def.is_regular()) { set_clustered_cell(prefix, def, std::move(value)); } else { throw std::runtime_error("attemting to store into a key cell"); } } std::experimental::optional mutation::get_cell(const clustering_key& rkey, const column_definition& def) const { if (def.is_static()) { const atomic_cell_or_collection* cell = _p.static_row().find_cell(def.id); if (!cell) { return {}; } return { *cell }; } else { const row* r = _p.find_row(rkey); if (!r) { return {}; } const atomic_cell_or_collection* cell = r->find_cell(def.id); return { *cell }; } } bool mutation::operator==(const mutation& m) const { return _dk.equal(*_schema, m._dk) && _p.equal(*_schema, m._p); } bool mutation::operator!=(const mutation& m) const { return !(*this == m); } query::result mutation::query(const query::partition_slice& slice, gc_clock::time_point now, uint32_t row_limit) const { query::result::builder builder(slice); auto pb = builder.add_partition(key()); _p.query(pb, *_schema, now, row_limit); return builder.build(); } size_t mutation::live_row_count(gc_clock::time_point query_time) const { return _p.live_row_count(*_schema, query_time); } bool mutation_decorated_key_less_comparator::operator()(const mutation& m1, const mutation& m2) const { return m1.decorated_key().less_compare(*m1.schema(), m2.decorated_key()); } boost::iterator_range::const_iterator> slice(const std::vector& partitions, const query::partition_range& r) { struct cmp { bool operator()(const dht::ring_position& pos, const mutation& m) const { return m.decorated_key().tri_compare(*m.schema(), pos) > 0; }; bool operator()(const mutation& m, const dht::ring_position& pos) const { return m.decorated_key().tri_compare(*m.schema(), pos) < 0; }; }; return boost::make_iterator_range( r.start() ? (r.start()->is_inclusive() ? std::lower_bound(partitions.begin(), partitions.end(), r.start()->value(), cmp()) : std::upper_bound(partitions.begin(), partitions.end(), r.start()->value(), cmp())) : partitions.cbegin(), r.end() ? (r.end()->is_inclusive() ? std::upper_bound(partitions.begin(), partitions.end(), r.end()->value(), cmp()) : std::lower_bound(partitions.begin(), partitions.end(), r.end()->value(), cmp())) : partitions.cend()); }