mirror of
https://github.com/0xbbjubjub/miniguest.git
synced 2025-01-07 12:30:53 +03:00
158 lines
5.0 KiB
C++
158 lines
5.0 KiB
C++
/* Copyright 2022 Julie Bettens
|
|
*
|
|
* This file is part of the Miniguest companion.
|
|
*
|
|
* The Miniguest companion 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.
|
|
*
|
|
* Miniguest 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 Miniguest. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "common.hpp"
|
|
#include "manifest.hpp"
|
|
|
|
#include "build-result.hh"
|
|
#include "command.hh"
|
|
#include "derived-path.hh"
|
|
#include "eval-cache.hh"
|
|
|
|
#include <iostream>
|
|
#include <map>
|
|
|
|
using namespace nix;
|
|
namespace fs = std::filesystem;
|
|
|
|
namespace miniguest {
|
|
|
|
struct CmdCreate : virtual EvalCommand, virtual MixProfile {
|
|
std::string guest_name;
|
|
std::optional<std::string> hypervisor;
|
|
|
|
const std::string set_color_red = "\e[1m\e[31m", reset_color = "\e(B\e[m";
|
|
|
|
std::string description() override {
|
|
return "generate a command to configure the guest in the hypervisor";
|
|
}
|
|
|
|
std::string doc() override {
|
|
return R""(miniguest create [-h|--help] [-t|--hypervisor <HYPERVISOR>] <guest-name>
|
|
<guest-name>: name of guest to create
|
|
-h, --help: Prints help
|
|
-t, --hypervisor: hypervisor to configure. Can be one of: 'libvirt' and 'lxc' (default: 'libvirt'))"";
|
|
}
|
|
|
|
GuestConfig load_config() {
|
|
auto path =
|
|
fs::path("/etc/miniguests") / guest_name / "miniguest-config.json";
|
|
try {
|
|
return nlohmann::json::parse(readFile(path)).get<GuestConfig>();
|
|
} catch (SysError &err) { logWarning(err.info()); }
|
|
warn("metadata unavailable: falling back to defaults");
|
|
return {};
|
|
}
|
|
|
|
void display_command(Strings cmd) const {
|
|
std::cout << "# Create " << guest_name << " using:" << std::endl;
|
|
for (auto &line : cmd)
|
|
std::cout << line << " \\\n ";
|
|
std::cout << std::endl;
|
|
}
|
|
|
|
void display_command_libvirt(const GuestConfig &cfg) const {
|
|
auto virtiofs = cfg.qemu_fs_type == QemuFsType::virtiofs;
|
|
std::string memorybacking =
|
|
virtiofs ? " --memorybacking=access.mode=shared" : "";
|
|
std::string fsArgs =
|
|
virtiofs ? "driver.type=virtiofs" : "readonly=yes,accessmode=squash";
|
|
Strings cmd{
|
|
"virt-install -n " + guest_name,
|
|
"--connect qemu:///system",
|
|
"--os-variant nixos-unstable",
|
|
"--memory 1536" + memorybacking,
|
|
"--disk none",
|
|
"--import",
|
|
"--boot kernel=/etc/miniguests/" + guest_name +
|
|
"/kernel,initrd=/etc/miniguests/" + guest_name + "/initrd",
|
|
"--filesystem /nix/store/,nix-store," + fsArgs,
|
|
"--filesystem /etc/miniguests/" + guest_name + "/boot/,boot," + fsArgs,
|
|
};
|
|
display_command(cmd);
|
|
}
|
|
|
|
void display_command_lxc(const GuestConfig &cfg) const {
|
|
Strings cmd = {
|
|
"lxc-create " + guest_name,
|
|
"-f extra-config",
|
|
"-t local --",
|
|
"-m @lxc_template@/meta.tar.xz",
|
|
"-f @lxc_template@/rootfs.tar.xz",
|
|
};
|
|
std::cout << "# " << set_color_red
|
|
<< "WARNING: make sure root is uid-mapped, otherwise you might "
|
|
"experience store corruption in the host!"
|
|
<< reset_color << std::endl;
|
|
display_command(cmd);
|
|
}
|
|
|
|
struct HypervisorData {
|
|
void (CmdCreate::*display_commmand)(const GuestConfig &cfg) const;
|
|
};
|
|
|
|
static const std::map<std::string, HypervisorData> hypervisors;
|
|
|
|
static void completeHypervisor(size_t, std::string_view prefix) {
|
|
completionType = ctNormal;
|
|
for (auto [name, _] : hypervisors)
|
|
if (name.size() >= prefix.size() &&
|
|
std::string_view(name.data(), prefix.size()) == prefix)
|
|
completions->add(name);
|
|
}
|
|
|
|
CmdCreate() {
|
|
expectArgs({
|
|
.label = "guest-name",
|
|
.handler = {&guest_name},
|
|
.completer = completeGuestName,
|
|
});
|
|
addFlag({
|
|
.longName = "hypervisor",
|
|
.shortName = 't',
|
|
.description = "hypervisor to configure (default: libvirt)",
|
|
.labels = {"hypervisor"},
|
|
.handler = {&hypervisor},
|
|
.completer = &completeHypervisor,
|
|
});
|
|
}
|
|
|
|
void run(ref<Store> store) override {
|
|
auto cfg = load_config();
|
|
|
|
if (!hypervisor)
|
|
hypervisor = cfg.guest_type == GuestType::lxc ? "lxc" : "libvirt";
|
|
|
|
if (auto it = hypervisors.find(*hypervisor); it != hypervisors.end()) {
|
|
(this->*it->second.display_commmand)(cfg);
|
|
} else
|
|
throw Error(2, "unknown hypervisor type: " + *hypervisor);
|
|
}
|
|
|
|
virtual ~CmdCreate() = default;
|
|
};
|
|
|
|
const std::map<std::string, CmdCreate::HypervisorData> CmdCreate::hypervisors{
|
|
{"libvirt", {&CmdCreate::display_command_libvirt}},
|
|
{"lxc", {&CmdCreate::display_command_lxc}},
|
|
};
|
|
|
|
static auto rCmdCreate = registerCommand<CmdCreate>("create");
|
|
|
|
} // namespace miniguest
|