// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*- // vim: ts=8 sw=2 smarttab ft=cpp #include "rgw_service.h" #include "services/svc_finisher.h" #include "services/svc_bi_rados.h" #include "services/svc_bilog_rados.h" #include "services/svc_bucket_sobj.h" #include "services/svc_bucket_sync_sobj.h" #include "services/svc_cls.h" #include "services/svc_config_key_rados.h" #include "services/svc_datalog_rados.h" #include "services/svc_mdlog.h" #include "services/svc_meta.h" #include "services/svc_meta_be.h" #include "services/svc_meta_be_sobj.h" #include "services/svc_meta_be_otp.h" #include "services/svc_notify.h" #include "services/svc_otp.h" #include "services/svc_rados.h" #include "services/svc_zone.h" #include "services/svc_zone_utils.h" #include "services/svc_quota.h" #include "services/svc_sync_modules.h" #include "services/svc_sys_obj.h" #include "services/svc_sys_obj_cache.h" #include "services/svc_sys_obj_core.h" #include "services/svc_user_rados.h" #include "common/errno.h" #include "rgw_metadata.h" #include "rgw_user.h" #include "rgw_bucket.h" #include "rgw_otp.h" #define dout_subsys ceph_subsys_rgw RGWServices_Def::RGWServices_Def() = default; RGWServices_Def::~RGWServices_Def() { shutdown(); } int RGWServices_Def::init(CephContext *cct, bool have_cache, bool raw, bool run_sync) { finisher = std::make_unique(cct); bucket_sobj = std::make_unique(cct); bucket_sync_sobj = std::make_unique(cct); bi_rados = std::make_unique(cct); bilog_rados = std::make_unique(cct); cls = std::make_unique(cct); config_key_rados = std::make_unique(cct); datalog_rados = std::make_unique(cct); mdlog = std::make_unique(cct, run_sync); meta = std::make_unique(cct); meta_be_sobj = std::make_unique(cct); meta_be_otp = std::make_unique(cct); notify = std::make_unique(cct); otp = std::make_unique(cct); rados = std::make_unique(cct); zone = std::make_unique(cct); zone_utils = std::make_unique(cct); quota = std::make_unique(cct); sync_modules = std::make_unique(cct); sysobj = std::make_unique(cct); sysobj_core = std::make_unique(cct); user_rados = std::make_unique(cct); if (have_cache) { sysobj_cache = std::make_unique(cct); } vector meta_bes{meta_be_sobj.get(), meta_be_otp.get()}; finisher->init(); bi_rados->init(zone.get(), rados.get(), bilog_rados.get(), datalog_rados.get()); bilog_rados->init(bi_rados.get()); bucket_sobj->init(zone.get(), sysobj.get(), sysobj_cache.get(), bi_rados.get(), meta.get(), meta_be_sobj.get(), sync_modules.get(), bucket_sync_sobj.get()); bucket_sync_sobj->init(zone.get(), sysobj.get(), sysobj_cache.get(), bucket_sobj.get()); cls->init(zone.get(), rados.get()); config_key_rados->init(rados.get()); datalog_rados->init(zone.get(), cls.get()); mdlog->init(rados.get(), zone.get(), sysobj.get(), cls.get()); meta->init(sysobj.get(), mdlog.get(), meta_bes); meta_be_sobj->init(sysobj.get(), mdlog.get()); meta_be_otp->init(sysobj.get(), mdlog.get(), cls.get()); notify->init(zone.get(), rados.get(), finisher.get()); otp->init(zone.get(), meta.get(), meta_be_otp.get()); rados->init(); zone->init(sysobj.get(), rados.get(), sync_modules.get(), bucket_sync_sobj.get()); zone_utils->init(rados.get(), zone.get()); quota->init(zone.get()); sync_modules->init(zone.get()); sysobj_core->core_init(rados.get(), zone.get()); if (have_cache) { sysobj_cache->init(rados.get(), zone.get(), notify.get()); sysobj->init(rados.get(), sysobj_cache.get()); } else { sysobj->init(rados.get(), sysobj_core.get()); } user_rados->init(rados.get(), zone.get(), sysobj.get(), sysobj_cache.get(), meta.get(), meta_be_sobj.get(), sync_modules.get()); can_shutdown = true; int r = finisher->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start finisher service (" << cpp_strerror(-r) << dendl; return r; } if (!raw) { r = notify->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start notify service (" << cpp_strerror(-r) << dendl; return r; } } r = rados->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start rados service (" << cpp_strerror(-r) << dendl; return r; } if (!raw) { r = zone->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start zone service (" << cpp_strerror(-r) << dendl; return r; } r = mdlog->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start mdlog service (" << cpp_strerror(-r) << dendl; return r; } r = sync_modules->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start sync modules service (" << cpp_strerror(-r) << dendl; return r; } } r = cls->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start cls service (" << cpp_strerror(-r) << dendl; return r; } r = config_key_rados->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start config_key service (" << cpp_strerror(-r) << dendl; return r; } r = zone_utils->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start zone_utils service (" << cpp_strerror(-r) << dendl; return r; } r = quota->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start quota service (" << cpp_strerror(-r) << dendl; return r; } r = sysobj_core->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start sysobj_core service (" << cpp_strerror(-r) << dendl; return r; } if (have_cache) { r = sysobj_cache->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start sysobj_cache service (" << cpp_strerror(-r) << dendl; return r; } } r = sysobj->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start sysobj service (" << cpp_strerror(-r) << dendl; return r; } if (!raw) { r = datalog_rados->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start datalog_rados service (" << cpp_strerror(-r) << dendl; return r; } r = meta_be_sobj->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start meta_be_sobj service (" << cpp_strerror(-r) << dendl; return r; } r = meta->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start meta service (" << cpp_strerror(-r) << dendl; return r; } r = bucket_sobj->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start bucket service (" << cpp_strerror(-r) << dendl; return r; } r = bucket_sync_sobj->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start bucket_sync service (" << cpp_strerror(-r) << dendl; return r; } r = user_rados->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start user_rados service (" << cpp_strerror(-r) << dendl; return r; } r = otp->start(); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start otp service (" << cpp_strerror(-r) << dendl; return r; } } /* cache or core services will be started by sysobj */ return 0; } void RGWServices_Def::shutdown() { if (!can_shutdown) { return; } if (has_shutdown) { return; } datalog_rados->shutdown(); sysobj->shutdown(); sysobj_core->shutdown(); notify->shutdown(); if (sysobj_cache) { sysobj_cache->shutdown(); } quota->shutdown(); zone_utils->shutdown(); zone->shutdown(); rados->shutdown(); has_shutdown = true; } int RGWServices::do_init(CephContext *_cct, bool have_cache, bool raw, bool run_sync) { cct = _cct; int r = _svc.init(cct, have_cache, raw, run_sync); if (r < 0) { return r; } finisher = _svc.finisher.get(); bi_rados = _svc.bi_rados.get(); bi = bi_rados; bilog_rados = _svc.bilog_rados.get(); bucket_sobj = _svc.bucket_sobj.get(); bucket = bucket_sobj; bucket_sync_sobj = _svc.bucket_sync_sobj.get(); bucket_sync = bucket_sync_sobj; cls = _svc.cls.get(); config_key_rados = _svc.config_key_rados.get(); config_key = config_key_rados; datalog_rados = _svc.datalog_rados.get(); mdlog = _svc.mdlog.get(); meta = _svc.meta.get(); meta_be_sobj = _svc.meta_be_sobj.get(); meta_be_otp = _svc.meta_be_otp.get(); notify = _svc.notify.get(); otp = _svc.otp.get(); rados = _svc.rados.get(); zone = _svc.zone.get(); zone_utils = _svc.zone_utils.get(); quota = _svc.quota.get(); sync_modules = _svc.sync_modules.get(); sysobj = _svc.sysobj.get(); cache = _svc.sysobj_cache.get(); core = _svc.sysobj_core.get(); user = _svc.user_rados.get(); return 0; } int RGWServiceInstance::start() { if (start_state != StateInit) { return 0; } start_state = StateStarting;; /* setting started prior to do_start() on purpose so that circular references can call start() on each other */ int r = do_start(); if (r < 0) { return r; } start_state = StateStarted; return 0; } RGWCtlDef::RGWCtlDef() {} RGWCtlDef::~RGWCtlDef() {} RGWCtlDef::_meta::_meta() {} RGWCtlDef::_meta::~_meta() {} int RGWCtlDef::init(RGWServices& svc) { meta.mgr.reset(new RGWMetadataManager(svc.meta)); meta.user.reset(RGWUserMetaHandlerAllocator::alloc(svc.user)); auto sync_module = svc.sync_modules->get_sync_module(); if (sync_module) { meta.bucket.reset(sync_module->alloc_bucket_meta_handler()); meta.bucket_instance.reset(sync_module->alloc_bucket_instance_meta_handler()); } else { meta.bucket.reset(RGWBucketMetaHandlerAllocator::alloc()); meta.bucket_instance.reset(RGWBucketInstanceMetaHandlerAllocator::alloc()); } meta.otp.reset(RGWOTPMetaHandlerAllocator::alloc()); user.reset(new RGWUserCtl(svc.zone, svc.user, (RGWUserMetadataHandler *)meta.user.get())); bucket.reset(new RGWBucketCtl(svc.zone, svc.bucket, svc.bucket_sync, svc.bi)); otp.reset(new RGWOTPCtl(svc.zone, svc.otp)); RGWBucketMetadataHandlerBase *bucket_meta_handler = static_cast(meta.bucket.get()); RGWBucketInstanceMetadataHandlerBase *bi_meta_handler = static_cast(meta.bucket_instance.get()); bucket_meta_handler->init(svc.bucket, bucket.get()); bi_meta_handler->init(svc.zone, svc.bucket, svc.bi); RGWOTPMetadataHandlerBase *otp_handler = static_cast(meta.otp.get()); otp_handler->init(svc.zone, svc.meta_be_otp, svc.otp); user->init(bucket.get()); bucket->init(user.get(), (RGWBucketMetadataHandler *)bucket_meta_handler, (RGWBucketInstanceMetadataHandler *)bi_meta_handler, svc.datalog_rados->get_log()); otp->init((RGWOTPMetadataHandler *)meta.otp.get()); return 0; } int RGWCtl::init(RGWServices *_svc) { svc = _svc; cct = svc->cct; int r = _ctl.init(*svc); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start init ctls (" << cpp_strerror(-r) << dendl; return r; } meta.mgr = _ctl.meta.mgr.get(); meta.user = _ctl.meta.user.get(); meta.bucket = _ctl.meta.bucket.get(); meta.bucket_instance = _ctl.meta.bucket_instance.get(); meta.otp = _ctl.meta.otp.get(); user = _ctl.user.get(); bucket = _ctl.bucket.get(); otp = _ctl.otp.get(); r = meta.user->attach(meta.mgr); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start init meta.user ctl (" << cpp_strerror(-r) << dendl; return r; } r = meta.bucket->attach(meta.mgr); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start init meta.bucket ctl (" << cpp_strerror(-r) << dendl; return r; } r = meta.bucket_instance->attach(meta.mgr); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start init meta.bucket_instance ctl (" << cpp_strerror(-r) << dendl; return r; } r = meta.otp->attach(meta.mgr); if (r < 0) { ldout(cct, 0) << "ERROR: failed to start init otp ctl (" << cpp_strerror(-r) << dendl; return r; } return 0; }