// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*- // vim: ts=8 sw=2 smarttab ft=cpp /* * Ceph - scalable distributed file system * * Copyright (C) 2019 Red Hat, Inc. * * This is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software * Foundation. See file COPYING. * */ #pragma once #include "include/rados/librados.hpp" #include "common/RWLock.h" #include "rgw_metadata.h" #include "rgw_mdlog_types.h" #include "rgw_tools.h" #define META_LOG_OBJ_PREFIX "meta.log." class RGWSI_Cls; class RGWSI_Zone; struct RGWMetadataLogInfo { std::string marker; real_time last_update; void dump(Formatter *f) const; void decode_json(JSONObj *obj); }; class RGWCompletionManager; class RGWMetadataLogInfoCompletion : public RefCountedObject { public: using info_callback_t = std::function; private: cls_log_header header; rgw_rados_ref io_obj; librados::AioCompletion *completion; std::mutex mutex; //< protects callback between cancel/complete boost::optional callback; //< cleared on cancel public: explicit RGWMetadataLogInfoCompletion(info_callback_t callback); ~RGWMetadataLogInfoCompletion() override; rgw_rados_ref& get_io_obj() { return io_obj; } cls_log_header& get_header() { return header; } librados::AioCompletion* get_completion() { return completion; } void finish(librados::completion_t cb) { std::lock_guard lock(mutex); if (callback) { (*callback)(completion->get_return_value(), header); } } void cancel() { std::lock_guard lock(mutex); callback = boost::none; } }; class RGWMetadataLog { CephContext *cct; const std::string prefix; struct Svc { RGWSI_Zone *zone{nullptr}; RGWSI_Cls *cls{nullptr}; } svc; static std::string make_prefix(const std::string& period) { if (period.empty()) return META_LOG_OBJ_PREFIX; return META_LOG_OBJ_PREFIX + period + "."; } RWLock lock; std::set modified_shards; void mark_modified(int shard_id); public: RGWMetadataLog(CephContext *_cct, RGWSI_Zone *_zone_svc, RGWSI_Cls *_cls_svc, const std::string& period) : cct(_cct), prefix(make_prefix(period)), lock("RGWMetaLog::lock") { svc.zone = _zone_svc; svc.cls = _cls_svc; } void get_shard_oid(int id, std::string& oid) const { char buf[16]; snprintf(buf, sizeof(buf), "%d", id); oid = prefix + buf; } int add_entry(const DoutPrefixProvider *dpp, const std::string& hash_key, const std::string& section, const std::string& key, bufferlist& bl, optional_yield y); int get_shard_id(const std::string& hash_key, int *shard_id); int store_entries_in_shard(const DoutPrefixProvider *dpp, std::list& entries, int shard_id, librados::AioCompletion *completion); struct LogListCtx { int cur_shard; std::string marker; real_time from_time; real_time end_time; std::string cur_oid; bool done; LogListCtx() : cur_shard(0), done(false) {} }; void init_list_entries(int shard_id, const real_time& from_time, const real_time& end_time, const std::string& marker, void **handle); void complete_list_entries(void *handle); int list_entries(const DoutPrefixProvider *dpp, void *handle, int max_entries, std::list& entries, std::string *out_marker, bool *truncated, optional_yield y); int trim(const DoutPrefixProvider *dpp, int shard_id, const real_time& from_time, const real_time& end_time, const std::string& start_marker, const std::string& end_marker, optional_yield y); int get_info(const DoutPrefixProvider *dpp, int shard_id, RGWMetadataLogInfo *info, optional_yield y); int get_info_async(const DoutPrefixProvider *dpp, int shard_id, RGWMetadataLogInfoCompletion *completion); int lock_exclusive(const DoutPrefixProvider *dpp, int shard_id, timespan duration, std::string&zone_id, std::string& owner_id); int unlock(const DoutPrefixProvider *dpp, int shard_id, std::string& zone_id, std::string& owner_id); int update_shards(std::list& shards); void read_clear_modified(std::set &modified); }; struct LogStatusDump { RGWMDLogStatus status; explicit LogStatusDump(RGWMDLogStatus _status) : status(_status) {} void dump(Formatter *f) const; }; struct RGWMetadataLogData { obj_version read_version; obj_version write_version; RGWMDLogStatus status; RGWMetadataLogData() : status(MDLOG_STATUS_UNKNOWN) {} void encode(bufferlist& bl) const; void decode(bufferlist::const_iterator& bl); void dump(Formatter *f) const; void decode_json(JSONObj *obj); static void generate_test_instances(std::list& l); }; WRITE_CLASS_ENCODER(RGWMetadataLogData) struct RGWMetadataLogHistory { epoch_t oldest_realm_epoch; std::string oldest_period_id; void encode(bufferlist& bl) const { ENCODE_START(1, 1, bl); encode(oldest_realm_epoch, bl); encode(oldest_period_id, bl); ENCODE_FINISH(bl); } void decode(bufferlist::const_iterator& p) { DECODE_START(1, p); decode(oldest_realm_epoch, p); decode(oldest_period_id, p); DECODE_FINISH(p); } static const std::string oid; }; WRITE_CLASS_ENCODER(RGWMetadataLogHistory)