/* Copyright © 2005-2013 Roger Leigh * * schroot is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * schroot is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see * . * *********************************************************************/ #include #include "chroot.h" #include "chroot-facet-mountable.h" #include "chroot-facet-session.h" #include "chroot-facet-session-clonable.h" #include "chroot-facet-source-clonable.h" #include "chroot-plain.h" #ifdef SBUILD_FEATURE_BLOCKDEV #include "chroot-block-device-base.h" #endif #ifdef SBUILD_FEATURE_LVMSNAP #include "chroot-lvm-snapshot.h" #endif // SBUILD_FEATURE_LVMSNAP #ifdef SBUILD_FEATURE_LOOPBACK #include "chroot-loopback.h" #endif // SBUILD_FEATURE_LOOPBACK #ifdef SBUILD_FEATURE_BTRFSSNAP #include "chroot-btrfs-snapshot.h" #endif // SBUILD_FEATURE_BTRFSSNAP #ifdef SBUILD_FEATURE_UNION #include "chroot-facet-union.h" #endif // SBUILD_FEATURE_UNION #include "format-detail.h" #include #include using boost::format; using std::endl; using namespace sbuild; chroot_facet_session_clonable::chroot_facet_session_clonable (): chroot_facet() { } chroot_facet_session_clonable::~chroot_facet_session_clonable () { } chroot_facet_session_clonable::ptr chroot_facet_session_clonable::create () { return ptr(new chroot_facet_session_clonable()); } chroot_facet::ptr chroot_facet_session_clonable::clone () const { return ptr(new chroot_facet_session_clonable(*this)); } std::string const& chroot_facet_session_clonable::get_name () const { static const std::string name("session"); return name; } void chroot_facet_session_clonable::clone_session_setup (chroot const& parent, chroot::ptr& clone, std::string const& session_id, std::string const& alias, std::string const& user, bool root) const { // Disable session cloning. clone->remove_facet(); // Disable source cloning. clone->remove_facet(); clone->add_facet(chroot_facet_session::create()); // Disable session, delete aliases. chroot_facet_session::ptr psess(clone->get_facet()); assert(psess); psess->set_original_name(clone->get_name()); psess->set_selected_name(alias); clone->set_name(session_id); assert(clone->get_name() == session_id); clone->set_description (clone->get_description() + ' ' + _("(session chroot)")); string_list empty_list; string_list allowed_users; if (!user.empty()) allowed_users.push_back(user); if (root) { clone->set_users(empty_list); clone->set_root_users(allowed_users); } else { clone->set_users(allowed_users); clone->set_root_users(empty_list); } clone->set_groups(empty_list); clone->set_root_groups(empty_list); clone->set_aliases(empty_list); log_debug(DEBUG_INFO) << format("Cloned session %1%") % clone->get_name() << endl; /* If a chroot mount location has not yet been set, and the chroot is not a plain chroot, set a mount location with the session id. Only set for non-plain chroots which run setup scripts. */ { std::shared_ptr plain(std::dynamic_pointer_cast(clone)); if (clone->get_mount_location().empty() && !plain) { log_debug(DEBUG_NOTICE) << "Setting mount location" << endl; std::string location(std::string(SCHROOT_MOUNT_DIR) + "/" + session_id); clone->set_mount_location(location); } } log_debug(DEBUG_NOTICE) << format("Mount Location: %1%") % clone->get_mount_location() << endl; #ifdef SBUILD_FEATURE_BLOCKDEV /* Block devices need the mount device name specifying. */ /* Note that this will be overridden by LVM snapshot, below, so the order here is important. */ std::shared_ptr blockdevbase(std::dynamic_pointer_cast(clone)); if (blockdevbase) { chroot_facet_mountable::ptr pmnt (clone->get_facet()); if (pmnt) pmnt->set_mount_device(blockdevbase->get_device()); } #endif // SBUILD_FEATURE_BLOCKDEV #ifdef SBUILD_FEATURE_LOOPBACK /* Loopback chroots need the mount device name specifying. */ std::shared_ptr loopback(std::dynamic_pointer_cast(clone)); if (loopback) { chroot_facet_mountable::ptr pmnt (clone->get_facet()); if (pmnt) pmnt->set_mount_device(loopback->get_file()); } #endif // SBUILD_FEATURE_LOOPBACK #ifdef SBUILD_FEATURE_LVMSNAP /* LVM devices need the snapshot device name specifying. */ std::shared_ptr snapshot(std::dynamic_pointer_cast(clone)); if (snapshot && !snapshot->get_device().empty()) { std::string device(dirname(snapshot->get_device())); device += "/" + clone->get_name(); snapshot->set_snapshot_device(device); } #endif // SBUILD_FEATURE_LVMSNAP #ifdef SBUILD_FEATURE_BTRFSSNAP /* Btrfs snapshots need the snapshot name specifying. */ std::shared_ptr btrfs_snapshot(std::dynamic_pointer_cast(clone)); if (btrfs_snapshot && !btrfs_snapshot->get_snapshot_directory().empty()) { std::string snapname(btrfs_snapshot->get_snapshot_directory()); snapname += "/" + clone->get_name(); btrfs_snapshot->set_snapshot_name(snapname); } #endif // SBUILD_FEATURE_BTRFSSNAP #ifdef SBUILD_FEATURE_UNION // If the parent did not have a union facet, then neither should we. chroot_facet_union::const_ptr pparentuni(parent.get_facet()); if (!pparentuni) clone->remove_facet(); /* Filesystem unions need the overlay directory specifying. */ chroot_facet_union::ptr puni(clone->get_facet()); if (puni) { std::string overlay = puni->get_union_overlay_directory(); overlay += "/" + clone->get_name(); puni->set_union_overlay_directory(overlay); std::string underlay = puni->get_union_underlay_directory(); underlay += "/" + clone->get_name(); puni->set_union_underlay_directory(underlay); } #endif // SBUILD_FEATURE_UNION } void chroot_facet_session_clonable::setup_env (chroot const& chroot, environment& env) const { } sbuild::chroot::session_flags chroot_facet_session_clonable::get_session_flags (chroot const& chroot) const { return chroot::SESSION_CREATE; } void chroot_facet_session_clonable::get_details (chroot const& chroot, format_detail& detail) const { } void chroot_facet_session_clonable::get_used_keys (string_list& used_keys) const { } void chroot_facet_session_clonable::get_keyfile (chroot const& chroot, keyfile& keyfile) const { } void chroot_facet_session_clonable::set_keyfile (chroot& chroot, keyfile const& keyfile) { }