1 // -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
2 // vim: ts=8 sw=2 smarttab
4 * Ceph - scalable distributed file system
6 * Copyright (C) 2017 Red Hat Ltd
8 * This is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License version 2.1, as published by the Free Software
11 * Foundation. See file COPYING.
16 #include "OSDMonitor.h"
18 #include "FSCommands.h"
19 #include "MDSMonitor.h"
20 #include "MgrStatMonitor.h"
23 static const string
EXPERIMENTAL_WARNING("Warning! This feature is experimental."
24 "It may cause problems up to and including data loss."
25 "Consult the documentation at ceph.com, and if unsure, do not proceed."
26 "Add --yes-i-really-mean-it if you are certain.");
30 class FlagSetHandler
: public FileSystemCommandHandler
34 : FileSystemCommandHandler("fs flag set")
42 const cmdmap_t
& cmdmap
,
43 std::stringstream
&ss
) override
46 cmd_getval(g_ceph_context
, cmdmap
, "flag_name", flag_name
);
49 cmd_getval(g_ceph_context
, cmdmap
, "val", flag_val
);
52 cmd_getval(g_ceph_context
, cmdmap
, "yes_i_really_mean_it", sure
);
54 if (flag_name
== "enable_multiple") {
55 bool flag_bool
= false;
56 int r
= parse_bool(flag_val
, &flag_bool
, ss
);
58 ss
<< "Invalid boolean value '" << flag_val
<< "'";
62 bool jewel
= mon
->get_quorum_con_features() & CEPH_FEATURE_SERVER_JEWEL
;
63 if (flag_bool
&& !jewel
) {
64 ss
<< "Multiple-filesystems are forbidden until all mons are updated";
68 ss
<< EXPERIMENTAL_WARNING
;
70 fsmap
.set_enable_multiple(flag_bool
);
73 ss
<< "Unknown flag '" << flag_name
<< "'";
79 class FailHandler
: public FileSystemCommandHandler
83 : FileSystemCommandHandler("fs fail")
91 const cmdmap_t
& cmdmap
,
92 std::stringstream
& ss
) override
94 if (!mon
->osdmon()->is_writeable()) {
95 // not allowed to write yet, so retry when we can
96 mon
->osdmon()->wait_for_writeable(op
, new PaxosService::C_RetryMessage(mon
->mdsmon(), op
));
101 if (!cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
) || fs_name
.empty()) {
102 ss
<< "Missing filesystem name";
106 auto fs
= fsmap
.get_filesystem(fs_name
);
108 ss
<< "Not found: '" << fs_name
<< "'";
112 auto f
= [](auto fs
) {
113 fs
->mds_map
.set_flag(CEPH_MDSMAP_NOT_JOINABLE
);
115 fsmap
.modify_filesystem(fs
->fscid
, std::move(f
));
117 std::vector
<mds_gid_t
> to_fail
;
118 for (const auto& p
: fs
->mds_map
.get_mds_info()) {
119 to_fail
.push_back(p
.first
);
122 for (const auto& gid
: to_fail
) {
123 mon
->mdsmon()->fail_mds_gid(fsmap
, gid
);
125 if (!to_fail
.empty()) {
126 mon
->osdmon()->propose_pending();
130 ss
<< " marked not joinable; MDS cannot join the cluster. All MDS ranks marked failed.";
136 class FsNewHandler
: public FileSystemCommandHandler
139 explicit FsNewHandler(Paxos
*paxos
)
140 : FileSystemCommandHandler("fs new"), m_paxos(paxos
)
144 bool batched_propose() override
{
152 const cmdmap_t
& cmdmap
,
153 std::stringstream
&ss
) override
155 ceph_assert(m_paxos
->is_plugged());
157 string metadata_name
;
158 cmd_getval(g_ceph_context
, cmdmap
, "metadata", metadata_name
);
159 int64_t metadata
= mon
->osdmon()->osdmap
.lookup_pg_pool_name(metadata_name
);
161 ss
<< "pool '" << metadata_name
<< "' does not exist";
166 cmd_getval(g_ceph_context
, cmdmap
, "data", data_name
);
167 int64_t data
= mon
->osdmon()->osdmap
.lookup_pg_pool_name(data_name
);
169 ss
<< "pool '" << data_name
<< "' does not exist";
173 ss
<< "pool '" << data_name
<< "' has id 0, which CephFS does not allow. Use another pool or recreate it to get a non-zero pool id.";
178 cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
);
179 if (fs_name
.empty()) {
180 // Ensure fs name is not empty so that we can implement
181 // commmands that refer to FS by name in future.
182 ss
<< "Filesystem name may not be empty";
186 if (fsmap
.get_filesystem(fs_name
)) {
187 auto fs
= fsmap
.get_filesystem(fs_name
);
188 if (*(fs
->mds_map
.get_data_pools().begin()) == data
189 && fs
->mds_map
.get_metadata_pool() == metadata
) {
190 // Identical FS created already, this is a no-op
191 ss
<< "filesystem '" << fs_name
<< "' already exists";
194 ss
<< "filesystem already exists with name '" << fs_name
<< "'";
200 cmd_getval(g_ceph_context
,cmdmap
, "force", force
);
202 const pool_stat_t
*stat
= mon
->mgrstatmon()->get_pool_stat(metadata
);
204 int64_t metadata_num_objects
= stat
->stats
.sum
.num_objects
;
205 if (!force
&& metadata_num_objects
> 0) {
206 ss
<< "pool '" << metadata_name
207 << "' already contains some objects. Use an empty pool instead.";
212 if (fsmap
.filesystem_count() > 0
213 && !fsmap
.get_enable_multiple()) {
214 ss
<< "Creation of multiple filesystems is disabled. To enable "
215 "this experimental feature, use 'ceph fs flag set enable_multiple "
220 for (auto& fs
: fsmap
.get_filesystems()) {
221 const std::vector
<int64_t> &data_pools
= fs
->mds_map
.get_data_pools();
224 cmd_getval(g_ceph_context
, cmdmap
,
225 "allow_dangerous_metadata_overlay", sure
);
227 if ((std::find(data_pools
.begin(), data_pools
.end(), data
) != data_pools
.end()
228 || fs
->mds_map
.get_metadata_pool() == metadata
)
230 ss
<< "Filesystem '" << fs_name
231 << "' is already using one of the specified RADOS pools. This should ONLY be done in emergencies and after careful reading of the documentation. Pass --allow-dangerous-metadata-overlay to permit this.";
236 pg_pool_t
const *data_pool
= mon
->osdmon()->osdmap
.get_pg_pool(data
);
237 ceph_assert(data_pool
!= NULL
); // Checked it existed above
238 pg_pool_t
const *metadata_pool
= mon
->osdmon()->osdmap
.get_pg_pool(metadata
);
239 ceph_assert(metadata_pool
!= NULL
); // Checked it existed above
241 int r
= _check_pool(mon
->osdmon()->osdmap
, data
, false, force
, &ss
);
246 r
= _check_pool(mon
->osdmon()->osdmap
, metadata
, true, force
, &ss
);
251 if (!mon
->osdmon()->is_writeable()) {
252 // not allowed to write yet, so retry when we can
253 mon
->osdmon()->wait_for_writeable(op
, new PaxosService::C_RetryMessage(mon
->mdsmon(), op
));
256 mon
->osdmon()->do_application_enable(data
,
257 pg_pool_t::APPLICATION_NAME_CEPHFS
,
259 mon
->osdmon()->do_application_enable(metadata
,
260 pg_pool_t::APPLICATION_NAME_CEPHFS
,
261 "metadata", fs_name
);
262 mon
->osdmon()->propose_pending();
264 // All checks passed, go ahead and create.
265 auto&& fs
= fsmap
.create_filesystem(fs_name
, metadata
, data
,
266 mon
->get_quorum_con_features());
268 ss
<< "new fs with metadata pool " << metadata
<< " and data pool " << data
;
270 // assign a standby to rank 0 to avoid health warnings
272 mds_gid_t gid
= fsmap
.find_replacement_for({fs
->fscid
, 0}, _name
);
274 if (gid
!= MDS_GID_NONE
) {
275 const auto &info
= fsmap
.get_info_gid(gid
);
276 mon
->clog
->info() << info
.human_name() << " assigned to filesystem "
277 << fs_name
<< " as rank 0";
278 fsmap
.promote(gid
, *fs
, 0);
288 class SetHandler
: public FileSystemCommandHandler
292 : FileSystemCommandHandler("fs set")
299 const cmdmap_t
& cmdmap
,
300 std::stringstream
&ss
) override
303 if (!cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
) || fs_name
.empty()) {
304 ss
<< "Missing filesystem name";
308 auto fs
= fsmap
.get_filesystem(fs_name
);
310 ss
<< "Not found: '" << fs_name
<< "'";
315 if (!cmd_getval(g_ceph_context
, cmdmap
, "var", var
) || var
.empty()) {
316 ss
<< "Invalid variable";
322 if (!cmd_getval(g_ceph_context
, cmdmap
, "val", val
)) {
325 // we got a string. see if it contains an int.
326 n
= strict_strtoll(val
.c_str(), 10, &interr
);
327 if (var
== "max_mds") {
328 // NOTE: see also "mds set_max_mds", which can modify the same field.
329 if (interr
.length()) {
335 ss
<< "You must specify at least one MDS";
338 if (n
> 1 && n
> fs
->mds_map
.get_max_mds()) {
339 if (fs
->mds_map
.was_snaps_ever_allowed() &&
340 !fs
->mds_map
.allows_multimds_snaps()) {
341 ss
<< "multi-active MDS is not allowed while there are snapshots possibly created by pre-mimic MDS";
346 ss
<< "may not have more than " << MAX_MDS
<< " MDS ranks";
350 fsmap
.modify_filesystem(
352 [n
](std::shared_ptr
<Filesystem
> fs
)
354 fs
->mds_map
.clear_flag(CEPH_MDSMAP_NOT_JOINABLE
);
355 fs
->mds_map
.set_max_mds(n
);
357 } else if (var
== "inline_data") {
358 bool enable_inline
= false;
359 int r
= parse_bool(val
, &enable_inline
, ss
);
365 bool confirm
= false;
366 cmd_getval(g_ceph_context
, cmdmap
, "yes_i_really_mean_it", confirm
);
368 ss
<< EXPERIMENTAL_WARNING
;
371 ss
<< "inline data enabled";
373 fsmap
.modify_filesystem(
375 [](std::shared_ptr
<Filesystem
> fs
)
377 fs
->mds_map
.set_inline_data_enabled(true);
381 CompatSet c
= fsmap
.get_compat();
382 c
.incompat
.insert(MDS_FEATURE_INCOMPAT_INLINE
);
383 fsmap
.update_compat(c
);
385 ss
<< "inline data disabled";
386 fsmap
.modify_filesystem(
388 [](std::shared_ptr
<Filesystem
> fs
)
390 fs
->mds_map
.set_inline_data_enabled(false);
393 } else if (var
== "balancer") {
395 ss
<< "unsetting the metadata load balancer";
397 ss
<< "setting the metadata load balancer to " << val
;
399 fsmap
.modify_filesystem(
401 [val
](std::shared_ptr
<Filesystem
> fs
)
403 fs
->mds_map
.set_balancer(val
);
406 } else if (var
== "max_file_size") {
407 if (interr
.length()) {
408 ss
<< var
<< " requires an integer value";
411 if (n
< CEPH_MIN_STRIPE_UNIT
) {
412 ss
<< var
<< " must at least " << CEPH_MIN_STRIPE_UNIT
;
415 fsmap
.modify_filesystem(
417 [n
](std::shared_ptr
<Filesystem
> fs
)
419 fs
->mds_map
.set_max_filesize(n
);
421 } else if (var
== "allow_new_snaps") {
422 bool enable_snaps
= false;
423 int r
= parse_bool(val
, &enable_snaps
, ss
);
429 fsmap
.modify_filesystem(
431 [](std::shared_ptr
<Filesystem
> fs
)
433 fs
->mds_map
.clear_snaps_allowed();
435 ss
<< "disabled new snapshots";
437 fsmap
.modify_filesystem(
439 [](std::shared_ptr
<Filesystem
> fs
)
441 fs
->mds_map
.set_snaps_allowed();
443 ss
<< "enabled new snapshots";
445 } else if (var
== "allow_multimds") {
446 ss
<< "Multiple MDS is always enabled. Use the max_mds"
447 << " parameter to control the number of active MDSs"
448 << " allowed. This command is DEPRECATED and will be"
449 << " REMOVED from future releases.";
450 } else if (var
== "allow_multimds_snaps") {
452 int r
= parse_bool(val
, &enable
, ss
);
458 if (!cmd_getval(g_ceph_context
, cmdmap
, "confirm", confirm
) ||
459 confirm
!= "--yes-i-am-really-a-mds") {
460 ss
<< "Warning! This command is for MDS only. Do not run it manually";
465 ss
<< "enabled multimds with snapshot";
466 fsmap
.modify_filesystem(
468 [](std::shared_ptr
<Filesystem
> fs
)
470 fs
->mds_map
.set_multimds_snaps_allowed();
473 ss
<< "disabled multimds with snapshot";
474 fsmap
.modify_filesystem(
476 [](std::shared_ptr
<Filesystem
> fs
)
478 fs
->mds_map
.clear_multimds_snaps_allowed();
481 } else if (var
== "allow_dirfrags") {
482 ss
<< "Directory fragmentation is now permanently enabled."
483 << " This command is DEPRECATED and will be REMOVED from future releases.";
484 } else if (var
== "down") {
485 bool is_down
= false;
486 int r
= parse_bool(val
, &is_down
, ss
);
491 ss
<< fs
->mds_map
.get_fs_name();
493 fsmap
.modify_filesystem(
495 [is_down
](std::shared_ptr
<Filesystem
> fs
)
498 if (fs
->mds_map
.get_max_mds() > 0) {
499 fs
->mds_map
.set_old_max_mds();
500 fs
->mds_map
.set_max_mds(0);
501 } /* else already down! */
503 mds_rank_t oldmax
= fs
->mds_map
.get_old_max_mds();
504 fs
->mds_map
.set_max_mds(oldmax
? oldmax
: 1);
509 ss
<< " marked down. ";
511 ss
<< " marked up, max_mds = " << fs
->mds_map
.get_max_mds();
513 } else if (var
== "cluster_down" || var
== "joinable") {
514 bool joinable
= true;
515 int r
= parse_bool(val
, &joinable
, ss
);
519 if (var
== "cluster_down") {
520 joinable
= !joinable
;
523 ss
<< fs
->mds_map
.get_fs_name();
525 fsmap
.modify_filesystem(
527 [joinable
](std::shared_ptr
<Filesystem
> fs
)
530 fs
->mds_map
.clear_flag(CEPH_MDSMAP_NOT_JOINABLE
);
532 fs
->mds_map
.set_flag(CEPH_MDSMAP_NOT_JOINABLE
);
537 ss
<< " marked joinable; MDS may join as newly active.";
539 ss
<< " marked not joinable; MDS cannot join as newly active.";
542 if (var
== "cluster_down") {
543 ss
<< " WARNING: cluster_down flag is deprecated and will be"
544 << " removed in a future version. Please use \"joinable\".";
546 } else if (var
== "standby_count_wanted") {
547 if (interr
.length()) {
548 ss
<< var
<< " requires an integer value";
552 ss
<< var
<< " must be non-negative";
555 fsmap
.modify_filesystem(
557 [n
](std::shared_ptr
<Filesystem
> fs
)
559 fs
->mds_map
.set_standby_count_wanted(n
);
561 } else if (var
== "session_timeout") {
562 if (interr
.length()) {
563 ss
<< var
<< " requires an integer value";
567 ss
<< var
<< " must be at least 30s";
570 fsmap
.modify_filesystem(
572 [n
](std::shared_ptr
<Filesystem
> fs
)
574 fs
->mds_map
.set_session_timeout((uint32_t)n
);
576 } else if (var
== "session_autoclose") {
577 if (interr
.length()) {
578 ss
<< var
<< " requires an integer value";
582 ss
<< var
<< " must be at least 30s";
585 fsmap
.modify_filesystem(
587 [n
](std::shared_ptr
<Filesystem
> fs
)
589 fs
->mds_map
.set_session_autoclose((uint32_t)n
);
591 } else if (var
== "allow_standby_replay") {
593 int r
= parse_bool(val
, &allow
, ss
);
598 auto f
= [allow
](auto& fs
) {
600 fs
->mds_map
.set_standby_replay_allowed();
602 fs
->mds_map
.clear_standby_replay_allowed();
605 fsmap
.modify_filesystem(fs
->fscid
, std::move(f
));
606 } else if (var
== "min_compat_client") {
607 int vno
= ceph_release_from_name(val
.c_str());
609 ss
<< "version " << val
<< " is not recognized";
612 fsmap
.modify_filesystem(
614 [vno
](std::shared_ptr
<Filesystem
> fs
)
616 fs
->mds_map
.set_min_compat_client((uint8_t)vno
);
619 ss
<< "unknown variable " << var
;
627 class AddDataPoolHandler
: public FileSystemCommandHandler
630 explicit AddDataPoolHandler(Paxos
*paxos
)
631 : FileSystemCommandHandler("fs add_data_pool"), m_paxos(paxos
)
634 bool batched_propose() override
{
642 const cmdmap_t
& cmdmap
,
643 std::stringstream
&ss
) override
645 ceph_assert(m_paxos
->is_plugged());
648 cmd_getval(g_ceph_context
, cmdmap
, "pool", poolname
);
651 if (!cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
)
652 || fs_name
.empty()) {
653 ss
<< "Missing filesystem name";
657 auto fs
= fsmap
.get_filesystem(fs_name
);
659 ss
<< "Not found: '" << fs_name
<< "'";
663 int64_t poolid
= mon
->osdmon()->osdmap
.lookup_pg_pool_name(poolname
);
666 poolid
= strict_strtol(poolname
.c_str(), 10, &err
);
668 ss
<< "pool '" << poolname
<< "' does not exist";
673 int r
= _check_pool(mon
->osdmon()->osdmap
, poolid
, false, false, &ss
);
678 // no-op when the data_pool already on fs
679 if (fs
->mds_map
.is_data_pool(poolid
)) {
680 ss
<< "data pool " << poolid
<< " is already on fs " << fs_name
;
684 if (!mon
->osdmon()->is_writeable()) {
685 // not allowed to write yet, so retry when we can
686 mon
->osdmon()->wait_for_writeable(op
, new PaxosService::C_RetryMessage(mon
->mdsmon(), op
));
689 mon
->osdmon()->do_application_enable(poolid
,
690 pg_pool_t::APPLICATION_NAME_CEPHFS
,
692 mon
->osdmon()->propose_pending();
694 fsmap
.modify_filesystem(
696 [poolid
](std::shared_ptr
<Filesystem
> fs
)
698 fs
->mds_map
.add_data_pool(poolid
);
701 ss
<< "added data pool " << poolid
<< " to fsmap";
710 class SetDefaultHandler
: public FileSystemCommandHandler
714 : FileSystemCommandHandler("fs set-default")
721 const cmdmap_t
& cmdmap
,
722 std::stringstream
&ss
) override
725 cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
);
726 auto fs
= fsmap
.get_filesystem(fs_name
);
728 ss
<< "filesystem '" << fs_name
<< "' does not exist";
732 fsmap
.set_legacy_client_fscid(fs
->fscid
);
737 class RemoveFilesystemHandler
: public FileSystemCommandHandler
740 RemoveFilesystemHandler()
741 : FileSystemCommandHandler("fs rm")
748 const cmdmap_t
& cmdmap
,
749 std::stringstream
&ss
) override
751 /* We may need to blacklist ranks. */
752 if (!mon
->osdmon()->is_writeable()) {
753 // not allowed to write yet, so retry when we can
754 mon
->osdmon()->wait_for_writeable(op
, new PaxosService::C_RetryMessage(mon
->mdsmon(), op
));
758 // Check caller has correctly named the FS to delete
759 // (redundant while there is only one FS, but command
760 // syntax should apply to multi-FS future)
762 cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
);
763 auto fs
= fsmap
.get_filesystem(fs_name
);
765 // Consider absence success to make deletes idempotent
766 ss
<< "filesystem '" << fs_name
<< "' does not exist";
770 // Check that no MDS daemons are active
771 if (fs
->mds_map
.get_num_up_mds() > 0) {
772 ss
<< "all MDS daemons must be inactive/failed before removing filesystem. See `ceph fs fail`.";
776 // Check for confirmation flag
778 cmd_getval(g_ceph_context
, cmdmap
, "yes_i_really_mean_it", sure
);
780 ss
<< "this is a DESTRUCTIVE operation and will make data in your filesystem permanently" \
781 " inaccessible. Add --yes-i-really-mean-it if you are sure you wish to continue.";
785 if (fsmap
.get_legacy_client_fscid() == fs
->fscid
) {
786 fsmap
.set_legacy_client_fscid(FS_CLUSTER_ID_NONE
);
789 std::vector
<mds_gid_t
> to_fail
;
790 // There may be standby_replay daemons left here
791 for (const auto &i
: fs
->mds_map
.get_mds_info()) {
792 ceph_assert(i
.second
.state
== MDSMap::STATE_STANDBY_REPLAY
);
793 to_fail
.push_back(i
.first
);
796 for (const auto &gid
: to_fail
) {
797 // Standby replays don't write, so it isn't important to
798 // wait for an osdmap propose here: ignore return value.
799 mon
->mdsmon()->fail_mds_gid(fsmap
, gid
);
801 if (!to_fail
.empty()) {
802 mon
->osdmon()->propose_pending(); /* maybe new blacklists */
805 fsmap
.erase_filesystem(fs
->fscid
);
811 class ResetFilesystemHandler
: public FileSystemCommandHandler
814 ResetFilesystemHandler()
815 : FileSystemCommandHandler("fs reset")
822 const cmdmap_t
& cmdmap
,
823 std::stringstream
&ss
) override
826 cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
);
827 auto fs
= fsmap
.get_filesystem(fs_name
);
829 ss
<< "filesystem '" << fs_name
<< "' does not exist";
830 // Unlike fs rm, we consider this case an error
834 // Check that no MDS daemons are active
835 if (fs
->mds_map
.get_num_up_mds() > 0) {
836 ss
<< "all MDS daemons must be inactive before resetting filesystem: set the cluster_down flag"
837 " and use `ceph mds fail` to make this so";
841 // Check for confirmation flag
843 cmd_getval(g_ceph_context
, cmdmap
, "yes_i_really_mean_it", sure
);
845 ss
<< "this is a potentially destructive operation, only for use by experts in disaster recovery. "
846 "Add --yes-i-really-mean-it if you are sure you wish to continue.";
850 fsmap
.reset_filesystem(fs
->fscid
);
856 class RemoveDataPoolHandler
: public FileSystemCommandHandler
859 RemoveDataPoolHandler()
860 : FileSystemCommandHandler("fs rm_data_pool")
867 const cmdmap_t
& cmdmap
,
868 std::stringstream
&ss
) override
871 cmd_getval(g_ceph_context
, cmdmap
, "pool", poolname
);
874 if (!cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
)
875 || fs_name
.empty()) {
876 ss
<< "Missing filesystem name";
880 auto fs
= fsmap
.get_filesystem(fs_name
);
882 ss
<< "Not found: '" << fs_name
<< "'";
886 int64_t poolid
= mon
->osdmon()->osdmap
.lookup_pg_pool_name(poolname
);
889 poolid
= strict_strtol(poolname
.c_str(), 10, &err
);
891 ss
<< "pool '" << poolname
<< "' does not exist";
893 } else if (poolid
< 0) {
894 ss
<< "invalid pool id '" << poolid
<< "'";
899 ceph_assert(poolid
>= 0); // Checked by parsing code above
901 if (fs
->mds_map
.get_first_data_pool() == poolid
) {
902 ss
<< "cannot remove default data pool";
908 fsmap
.modify_filesystem(fs
->fscid
,
909 [&r
, poolid
](std::shared_ptr
<Filesystem
> fs
)
911 r
= fs
->mds_map
.remove_data_pool(poolid
);
914 // It was already removed, succeed in silence
917 // We removed it, succeed
918 ss
<< "removed data pool " << poolid
<< " from fsmap";
921 // Unexpected error, bubble up
928 * For commands with an alternative prefix
931 class AliasHandler
: public T
933 std::string alias_prefix
;
936 explicit AliasHandler(const std::string
&new_prefix
)
939 alias_prefix
= new_prefix
;
942 std::string
const &get_prefix() override
{return alias_prefix
;}
948 const cmdmap_t
& cmdmap
,
949 std::stringstream
&ss
) override
951 return T::handle(mon
, fsmap
, op
, cmdmap
, ss
);
956 std::list
<std::shared_ptr
<FileSystemCommandHandler
> >
957 FileSystemCommandHandler::load(Paxos
*paxos
)
959 std::list
<std::shared_ptr
<FileSystemCommandHandler
> > handlers
;
961 handlers
.push_back(std::make_shared
<SetHandler
>());
962 handlers
.push_back(std::make_shared
<FailHandler
>());
963 handlers
.push_back(std::make_shared
<FlagSetHandler
>());
964 handlers
.push_back(std::make_shared
<AddDataPoolHandler
>(paxos
));
965 handlers
.push_back(std::make_shared
<RemoveDataPoolHandler
>());
966 handlers
.push_back(std::make_shared
<FsNewHandler
>(paxos
));
967 handlers
.push_back(std::make_shared
<RemoveFilesystemHandler
>());
968 handlers
.push_back(std::make_shared
<ResetFilesystemHandler
>());
970 handlers
.push_back(std::make_shared
<SetDefaultHandler
>());
971 handlers
.push_back(std::make_shared
<AliasHandler
<SetDefaultHandler
> >(
977 int FileSystemCommandHandler::_check_pool(
979 const int64_t pool_id
,
982 std::stringstream
*ss
) const
984 ceph_assert(ss
!= NULL
);
986 const pg_pool_t
*pool
= osd_map
.get_pg_pool(pool_id
);
988 *ss
<< "pool id '" << pool_id
<< "' does not exist";
992 const string
& pool_name
= osd_map
.get_pool_name(pool_id
);
994 if (pool
->is_erasure() && metadata
) {
995 *ss
<< "pool '" << pool_name
<< "' (id '" << pool_id
<< "')"
996 << " is an erasure-coded pool. Use of erasure-coded pools"
997 << " for CephFS metadata is not permitted";
999 } else if (pool
->is_erasure() && !pool
->allows_ecoverwrites()) {
1000 // non-overwriteable EC pools are only acceptable with a cache tier overlay
1001 if (!pool
->has_tiers() || !pool
->has_read_tier() || !pool
->has_write_tier()) {
1002 *ss
<< "pool '" << pool_name
<< "' (id '" << pool_id
<< "')"
1003 << " is an erasure-coded pool, with no overwrite support";
1007 // That cache tier overlay must be writeback, not readonly (it's the
1008 // write operations like modify+truncate we care about support for)
1009 const pg_pool_t
*write_tier
= osd_map
.get_pg_pool(
1011 ceph_assert(write_tier
!= NULL
); // OSDMonitor shouldn't allow DNE tier
1012 if (write_tier
->cache_mode
== pg_pool_t::CACHEMODE_FORWARD
1013 || write_tier
->cache_mode
== pg_pool_t::CACHEMODE_READONLY
) {
1014 *ss
<< "EC pool '" << pool_name
<< "' has a write tier ("
1015 << osd_map
.get_pool_name(pool
->write_tier
)
1016 << ") that is configured "
1017 "to forward writes. Use a cache mode such as 'writeback' for "
1023 if (pool
->is_tier()) {
1024 *ss
<< " pool '" << pool_name
<< "' (id '" << pool_id
1025 << "') is already in use as a cache tier.";
1029 if (!force
&& !pool
->application_metadata
.empty() &&
1030 pool
->application_metadata
.count(
1031 pg_pool_t::APPLICATION_NAME_CEPHFS
) == 0) {
1032 *ss
<< " pool '" << pool_name
<< "' (id '" << pool_id
1033 << "') has a non-CephFS application enabled.";
1037 // Nothing special about this pool, so it is permissible