diff --git a/.gitignore b/.gitignore index bbecef7..8b37381 100644 --- a/.gitignore +++ b/.gitignore @@ -1,5 +1,5 @@ random/ -server/libvirt -server/.zig-cache -server/zig-out +server/zig-libvirt +server/.zig-cache/ +server/zig-out/ web/ diff --git a/flake.lock b/flake.lock index 70d8736..06edc22 100644 --- a/flake.lock +++ b/flake.lock @@ -36,13 +36,49 @@ "type": "github" } }, + "flake-utils_3": { + "inputs": { + "systems": "systems_3" + }, + "locked": { + "lastModified": 1710146030, + "narHash": "sha256-SZ5L6eA7HJ/nmkzGG7/ISclqe6oZdOZTNoesiInkXPQ=", + "owner": "numtide", + "repo": "flake-utils", + "rev": "b1d9ab70662946ef0850d488da1c9019f3a9752a", + "type": "github" + }, + "original": { + "owner": "numtide", + "repo": "flake-utils", + "type": "github" + } + }, + "flake-utils_4": { + "inputs": { + "systems": "systems_4" + }, + "locked": { + "lastModified": 1710146030, + "narHash": "sha256-SZ5L6eA7HJ/nmkzGG7/ISclqe6oZdOZTNoesiInkXPQ=", + "owner": "numtide", + "repo": "flake-utils", + "rev": "b1d9ab70662946ef0850d488da1c9019f3a9752a", + "type": "github" + }, + "original": { + "owner": "numtide", + "repo": "flake-utils", + "type": "github" + } + }, "nixpkgs": { "locked": { - "lastModified": 1719075281, - "narHash": "sha256-CyyxvOwFf12I91PBWz43iGT1kjsf5oi6ax7CrvaMyAo=", + "lastModified": 1719254875, + "narHash": "sha256-ECni+IkwXjusHsm9Sexdtq8weAq/yUyt1TWIemXt3Ko=", "owner": "nixos", "repo": "nixpkgs", - "rev": "a71e967ef3694799d0c418c98332f7ff4cc5f6af", + "rev": "2893f56de08021cffd9b6b6dfc70fd9ccd51eb60", "type": "github" }, "original": { @@ -85,6 +121,36 @@ } }, "nixpkgs_3": { + "locked": { + "lastModified": 1719231935, + "narHash": "sha256-jvgDRtw1+w9u5L2oo/2J7pJhPuJ0HL53NMq5ssxvonA=", + "owner": "nixos", + "repo": "nixpkgs", + "rev": "4cc2ade958e237ed32f5d67a25377084ac8f108b", + "type": "github" + }, + "original": { + "owner": "nixos", + "repo": "nixpkgs", + "type": "github" + } + }, + "nixpkgs_4": { + "locked": { + "lastModified": 1718622336, + "narHash": "sha256-lywfxWRBn+lwdKaBy5x5uTkbCcEPUonCn6bK8OQPsw4=", + "owner": "nixos", + "repo": "nixpkgs", + "rev": "d0fc4188d246ab953653f00e9ce0cf51d10d5eda", + "type": "github" + }, + "original": { + "owner": "nixos", + "repo": "nixpkgs", + "type": "github" + } + }, + "nixpkgs_5": { "locked": { "lastModified": 1718622336, "narHash": "sha256-lywfxWRBn+lwdKaBy5x5uTkbCcEPUonCn6bK8OQPsw4=", @@ -123,7 +189,8 @@ "flake-utils": "flake-utils", "nixpkgs": "nixpkgs", "nixvirt": "nixvirt", - "zig2nix": "zig2nix" + "zig-libvirt": "zig-libvirt", + "zig2nix": "zig2nix_2" } }, "systems": { @@ -156,10 +223,60 @@ "type": "github" } }, - "zig2nix": { + "systems_3": { + "locked": { + "lastModified": 1681028828, + "narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=", + "owner": "nix-systems", + "repo": "default", + "rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e", + "type": "github" + }, + "original": { + "owner": "nix-systems", + "repo": "default", + "type": "github" + } + }, + "systems_4": { + "locked": { + "lastModified": 1681028828, + "narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=", + "owner": "nix-systems", + "repo": "default", + "rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e", + "type": "github" + }, + "original": { + "owner": "nix-systems", + "repo": "default", + "type": "github" + } + }, + "zig-libvirt": { "inputs": { "flake-utils": "flake-utils_2", - "nixpkgs": "nixpkgs_3" + "nixpkgs": "nixpkgs_3", + "zig2nix": "zig2nix" + }, + "locked": { + "lastModified": 1719425163, + "narHash": "sha256-XP6oDHvaJUtItVlOT8ELI/9GkCsCu3rG0SlR5qwFuhg=", + "owner": "Nomkid", + "repo": "zig-libvirt", + "rev": "8eabdba79f72a46916a2e835643a802ab29bf682", + "type": "github" + }, + "original": { + "owner": "Nomkid", + "repo": "zig-libvirt", + "type": "github" + } + }, + "zig2nix": { + "inputs": { + "flake-utils": "flake-utils_3", + "nixpkgs": "nixpkgs_4" }, "locked": { "lastModified": 1719192015, @@ -174,6 +291,25 @@ "repo": "zig2nix", "type": "github" } + }, + "zig2nix_2": { + "inputs": { + "flake-utils": "flake-utils_4", + "nixpkgs": "nixpkgs_5" + }, + "locked": { + "lastModified": 1719364424, + "narHash": "sha256-EGpIcuGfpx+NQJGUho8vK5Q3TFHdaHtTLoqV/V+n4oY=", + "owner": "Cloudef", + "repo": "zig2nix", + "rev": "67d44febbeb1fad4c861a5c4d67a8f6719f7f763", + "type": "github" + }, + "original": { + "owner": "Cloudef", + "repo": "zig2nix", + "type": "github" + } } }, "root": "root", diff --git a/flake.nix b/flake.nix index fd5fc5b..d3b72f2 100644 --- a/flake.nix +++ b/flake.nix @@ -10,11 +10,10 @@ # }; nixvirt.url = "github:Nomkid/NixVirt"; zig2nix.url = "github:Cloudef/zig2nix"; - # symlink.url = "github:schuelermine/nix-symlink"; - # zig-libvirt.url = "github:Nomkid/zig-libvirt"; + zig-libvirt.url = "github:Nomkid/zig-libvirt"; }; - outputs = { self, nixpkgs, flake-utils, nixvirt, zig2nix }: + outputs = { self, nixpkgs, flake-utils, nixvirt, zig2nix, zig-libvirt }: flake-utils.lib.eachDefaultSystem (system: let pkgs = nixpkgs.legacyPackages.${system}; @@ -46,6 +45,8 @@ default = zig-env.app-no-root [] "zig build run -- \"$@\""; }; + packages.zig-libvirt = zig-libvirt.packages.${system}.default; + devShells.default = zig-env.mkShell { nativeBuildInputs = [ pkgs.libvirt.outPath ]; }; diff --git a/server/build.zig b/server/build.zig index ed4e33d..c3256ad 100644 --- a/server/build.zig +++ b/server/build.zig @@ -1,19 +1,10 @@ const std = @import("std"); -const Build = std.Build; -const Step = Build.Step; pub fn build(b: *std.Build) !void { const target = b.standardTargetOptions(.{}); const optimize = b.standardOptimizeOption(.{}); - const translate_c = b.addTranslateC(.{ - .root_source_file = b.path("libvirt/include/libvirt/libvirt.h"), - .target = target, - .optimize = optimize, - }); - translate_c.addIncludeDir("libvirt/include"); - - const output = b.addInstallFile(translate_c.getOutput(), "libvirt.zig"); + const libvirt = b.dependency("libvirt", .{}); const exe = b.addExecutable(.{ .name = "server", @@ -21,14 +12,7 @@ pub fn build(b: *std.Build) !void { .target = target, .optimize = optimize, }); - exe.step.dependOn(&output.step); - exe.linkLibC(); - exe.root_module.addIncludePath(b.path("libvirt/include")); - // exe.linkSystemLibrary("libvirt"); - exe.addLibraryPath(b.path("libvirt/lib")); - exe.addObjectFile(b.path("libvirt/lib/libvirt.so.0.10000.0")); - exe.addObjectFile(b.path("libvirt/lib/libvirt-admin.so.0.10000.0")); - exe.addObjectFile(b.path("libvirt/lib/libvirt-qemu.so.0.10000.0")); + exe.root_module.addImport("libvirt", libvirt.module("libvirt")); b.installArtifact(exe); const run_cmd = b.addRunArtifact(exe); @@ -39,36 +23,4 @@ pub fn build(b: *std.Build) !void { const run_step = b.step("run", "Run the app"); run_step.dependOn(&run_cmd.step); - - const libvirt_test = b.addTest(.{ - .root_source_file = b.path("src/libvirt.zig"), - .target = target, - .optimize = optimize, - .link_libc = true, - }); - const libvirt_connection_test = b.addTest(.{ - .root_source_file = b.path("src/libvirt-connection.zig"), - .target = target, - .optimize = optimize, - .link_libc = true, - }); - const libvirt_unit_tests = [_]*Step.Compile{ - libvirt_test, - libvirt_connection_test, - }; - for (libvirt_unit_tests) |tests| { - tests.root_module.addIncludePath(b.path("libvirt/include")); - tests.addLibraryPath(b.path("libvirt/lib")); - tests.addObjectFile(b.path("libvirt/lib/libvirt.so.0.10000.0")); - tests.addObjectFile(b.path("libvirt/lib/libvirt-admin.so.0.10000.0")); - tests.addObjectFile(b.path("libvirt/lib/libvirt-qemu.so.0.10000.0")); - } - - const run_libvirt_tests = [_]*Step.Run{ - b.addRunArtifact(libvirt_test), - b.addRunArtifact(libvirt_connection_test), - }; - - const test_step = b.step("test", "Run unit tests"); - for (run_libvirt_tests) |tests| test_step.dependOn(&tests.step); } diff --git a/server/build.zig.zon b/server/build.zig.zon index ea1b5d8..c0dcf53 100644 --- a/server/build.zig.zon +++ b/server/build.zig.zon @@ -31,6 +31,9 @@ // // actually used. // .lazy = false, //}, + .libvirt = .{ + .path = "zig-libvirt", + }, }, .paths = .{ diff --git a/server/src/libvirt-c.zig b/server/src/libvirt-c.zig deleted file mode 100644 index ff0aad8..0000000 --- a/server/src/libvirt-c.zig +++ /dev/null @@ -1,7 +0,0 @@ -pub const c = @cImport({ - @cInclude("libvirt/libvirt.h"); - @cInclude("libvirt/libvirt-admin.h"); - @cInclude("libvirt/libvirt-lxc.h"); - @cInclude("libvirt/libvirt-qemu.h"); - @cInclude("libvirt/virterror.h"); -}); diff --git a/server/src/libvirt-connection.zig b/server/src/libvirt-connection.zig deleted file mode 100644 index 65b7880..0000000 --- a/server/src/libvirt-connection.zig +++ /dev/null @@ -1,241 +0,0 @@ -const std = @import("std"); -const mem = std.mem; -const heap = std.heap; - -const h = @import("libvirt-helper.zig"); -const c = @import("libvirt-c.zig").c; -const err = @import("libvirt-error.zig"); -const Domain = @import("libvirt-domain.zig"); -const Pool = @import("libvirt-pool.zig"); -const VirError = err.VirError; -const Connection = @This(); - -ptr: c.virConnectPtr, -allocator: mem.Allocator, - -pub fn init(uri: []const u8, allocator: mem.Allocator) VirError!Connection { - const connection = c.virConnectOpenAuth(@ptrCast(uri), c.virConnectAuthPtrDefault, 0); - if (connection) |conn| return .{ - .ptr = conn, - .allocator = allocator, - } else return err.handleError(); -} - -pub fn deinit(self: *const Connection) void { - _ = c.virConnectClose(self.ptr); -} - -pub fn getURI(self: *const Connection) Error![]u8 { - const uri = c.virConnectGetURI(self.ptr); - defer std.c.free(uri); - const str = h.string(uri); - return if (str.len == 0) err.handleError() else try self.allocator.dupe(u8, str); -} -pub fn freeURI(self: *const Connection, uri: []u8) void { - self.allocator.free(uri); -} - -test "connection" { - const conn = try Connection.init("qemu:///system", std.testing.allocator); - defer conn.deinit(); - - const uri = try conn.getURI(); - defer conn.freeURI(uri); -} - -// pub fn createPool(self: *const Connection, xml: String, flags: []const Pool.CreateFlags) !Pool { -// const pool = c.virStoragePoolCreateXML(self.ptr, @ptrCast(xml.str), intFromFlags(Pool.CreateFlags, flags)); -// return if (pool) |p| Pool.init(p, self.allocator) else handleError(); -// } -// pub fn definePoolFlags(self: *const Connection, xml: String, flags: []const Pool.DefineFlags) !Pool { -// const pool = c.virStoragePoolCreateXML(self.ptr, @ptrCast(xml.str), intFromFlags(Pool.DefineFlags, flags)); -// return if (pool) |p| Pool.init(p, self.allocator) else handleError(); -// } - -pub const Error = mem.Allocator.Error || VirError; - -pub fn pools(self: *const Connection) Error![]Pool { - const list = try Pool.listAllStoragePools(self, &[_]Pool.ListFlags{}); - var l = try self.allocator.alloc(Pool, list.len); - for (list, 0..) |p, i| l[i] = Pool.init(p, self.allocator); - return l; -} -pub fn freePools(self: *const Connection, slice: []Pool) void { - // for (slice) |pool| pool.deinit(); - self.allocator.free(slice); -} -pub fn iteratePools(self: *const Connection) Error!PoolIterator { - return .{ - .allocator = self.allocator, - .values = try self.pools(), - }; -} -pub const PoolIterator = struct { - allocator: mem.Allocator, - values: []Pool, - index: usize = 0, - - pub fn deinit(it: *PoolIterator) void { - for (it.values) |v| v.free() catch @panic("pool free error"); - it.allocator.free(it.values); - } - pub fn next(it: *PoolIterator) ?Pool { - if (it.index >= it.values.len) return null; - const v = it.values[it.index]; - it.index += 1; - return v; - } - pub fn reset(it: *PoolIterator) void { - it.index = 0; - } -}; - -pub fn numOfDomains(self: *const Connection) VirError!u32 { - return Domain.numOfDomains(self); -} -pub fn numOfDefinedDomains(self: *const Connection) VirError!u32 { - return Domain.numOfDefinedDomains(self); -} - -// pub fn createDomain(self: *const Connection, xml: String, flags: []const Domain.CreateFlags) !Domain { -// const domain = c.virDomainCreateXML(self.ptr, @ptrCast(xml.str), intFromFlags(Domain.CreateFlags, flags)); -// return if (domain) |d| Domain.init(d, self.allocator) else handleError(); -// } -// pub fn defineDomain(self: *const Connection, xml: String) !Domain { -// const domain = c.virDomainDefineXML(self.ptr, @ptrCast(xml.str)); -// return if (domain) |d| Domain.init(d, self.allocator) else handleError(); -// } -// pub fn defineDomainFlags(self: *const Connection, xml: String, flags: []const Domain.DefineFlags) !Domain { -// const domain = c.virDomainDefineXMLFlags(self.ptr, @ptrCast(xml.str), intFromFlags(Domain.DefineFlags, flags)); -// return if (domain) |d| Domain.init(d, self.allocator) else handleError(); -// } - -pub fn createDomain(self: *const Connection, xml: h.String, flags: []const Domain.CreateFlags) VirError!Domain { - const ptr = try Domain.createXML(self, xml, flags); - return Domain.init(ptr, self.allocator); -} - -pub fn domains(self: *const Connection) Error![]Domain { - const list = try Domain.listAllDomains(self, &[_]Domain.ListFlags{}); - var l = try self.allocator.alloc(Domain, list.len); - for (list, 0..) |p, i| l[i] = Domain.init(p, self.allocator); - return l; -} -pub fn freeDomains(self: *const Connection, slice: []Domain) void { - // for (slice) |domain| domain.deinit(); - self.allocator.free(slice); -} -pub fn iterateDomains(self: *const Connection) Error!DomainIterator { - return .{ - .allocator = self.allocator, - .values = try self.domains(), - }; -} -pub const DomainIterator = struct { - allocator: mem.Allocator, - values: []Domain, - index: usize = 0, - - pub fn deinit(it: *DomainIterator) void { - for (it.values) |v| v.free() catch @panic("domain free error"); - it.allocator.free(it.values); - } - pub fn next(it: *DomainIterator) ?Domain { - if (it.index >= it.values.len) return null; - const v = it.values[it.index]; - it.index += 1; - return v; - } - pub fn reset(it: *DomainIterator) void { - it.index = 0; - } -}; - -test "domains" { - const conn = try Connection.init("qemu:///system", std.testing.allocator); - defer conn.deinit(); - - // Enumerate domains - const num_active = try conn.numOfDomains(); - const num_inactive = try conn.numOfDefinedDomains(); - _ = .{ num_active, num_inactive }; - - // Iterate domains - var domain_it = try conn.iterateDomains(); - defer domain_it.deinit(); - while (domain_it.next()) |domain| { - const active = try domain.isActive(); - const name = try domain.getName(); - _ = .{ active, name }; - } - - // Create and destory domain - const xml = try h.String.init(std.testing.allocator, - \\ - \\ vm1 - \\ 48dd0909-6346-4028-b922-1b3a13571360 - \\ 1048576 - \\ 1048576 - \\ 1 - \\ - \\ hvm - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ destroy - \\ restart - \\ destroy - \\ - \\ - \\ - \\ - \\ - \\ /run/libvirt/nix-emulators/qemu-system-x86_64 - \\ - \\
- \\ - \\ - \\ - \\
- \\ - \\ - \\ - \\
- \\ - \\ - \\ - \\
- \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\ - \\