clean up VLC-related code

This commit is contained in:
Jeeves 2025-02-15 06:14:29 -07:00
parent a7ca2f9377
commit 6ce0d30b4c

View file

@ -1,5 +1,4 @@
const std = @import("std"); const std = @import("std");
const xml = @import("./xml.zig");
// Music Player: // Music Player:
// - Launch VLC and control it via HTTP interface // - Launch VLC and control it via HTTP interface
@ -20,16 +19,8 @@ pub fn main() !void {
defer _ = gpa.deinit(); defer _ = gpa.deinit();
const allocator = gpa.allocator(); const allocator = gpa.allocator();
// var vlc = std.process.Child.init(&[_][]const u8{ var vlc = try VLC.init(allocator, try std.Uri.parse("http://localhost:8080"));
// "vlc", defer vlc.deinit();
// "--intf",
// "http",
// "--http-host",
// "localhost",
// "--http-password",
// "1234",
// }, allocator);
// try vlc.spawn();
var tz_file = try std.fs.openFileAbsolute("/etc/localtime", .{}); var tz_file = try std.fs.openFileAbsolute("/etc/localtime", .{});
defer tz_file.close(); defer tz_file.close();
@ -38,25 +29,24 @@ pub fn main() !void {
while (true) : (std.time.sleep(1_000_000_000)) { // sleep 500ms while (true) : (std.time.sleep(1_000_000_000)) { // sleep 500ms
try updateTime(tz); try updateTime(tz);
try updateStatus(allocator); try updateStatus(allocator, &vlc);
scroll += 1; scroll += 1;
} }
// try vlc.kill();
} }
const base64_encoder = std.base64.standard.Encoder; const base64_encoder = std.base64.standard.Encoder;
// TODO make the URL something short like jeevio.xyz/streamboy
const topic = "Something Fun For Everyone With Streamboy!!!!!!!! git.jeevio.xyz/jeeves/streamboy"; const topic = "Something Fun For Everyone With Streamboy!!!!!!!! git.jeevio.xyz/jeeves/streamboy";
var scroll: usize = 0; var scroll: usize = 0;
fn updateStatus(allocator: std.mem.Allocator) !void { fn updateStatus(allocator: std.mem.Allocator, vlc: *VLC) !void {
var stream_info_file = try std.fs.createFileAbsolute("/tmp/streaminfo", .{ .truncate = true }); var stream_info_file = try std.fs.createFileAbsolute("/tmp/streaminfo", .{ .truncate = true });
defer stream_info_file.close(); defer stream_info_file.close();
var song_info = try getSongInfo(allocator); var song_info = try vlc.getSongInfo();
defer song_info.deinit(allocator); defer song_info.deinit();
const string = try std.fmt.allocPrint(allocator, "{s} | ♪ {s} - {s} | ", .{ const string = try std.fmt.allocPrint(allocator, "{s} | ♪ {s} - {s} | ", .{
topic, topic,
@ -72,91 +62,154 @@ fn updateStatus(allocator: std.mem.Allocator) !void {
} }
} }
const SongInfo = struct { pub const VLC = struct {
title: ?[]const u8, allocator: std.mem.Allocator,
album: ?[]const u8, http: std.http.Client,
artist: ?[]const u8, base_uri: std.Uri,
authorization: []u8,
pub fn deinit(self: *SongInfo, allocator: std.mem.Allocator) void { pub fn init(allocator: std.mem.Allocator, base_uri: std.Uri) !VLC {
if (self.title) |b| allocator.free(b); const userpass = try std.fmt.allocPrint(allocator, ":{s}", .{"1234"});
if (self.album) |b| allocator.free(b); defer allocator.free(userpass);
if (self.artist) |b| allocator.free(b);
const base64_userpass = try allocator.alloc(u8, base64_encoder.calcSize(userpass.len));
defer allocator.free(base64_userpass);
const authorization = try std.fmt.allocPrint(allocator, "Basic {s}", .{base64_encoder.encode(base64_userpass, userpass)});
errdefer allocator.free(authorization);
// var vlc = std.process.Child.init(&[_][]const u8{
// "vlc",
// "--intf",
// "http",
// "--http-host",
// "localhost",
// "--http-password",
// "1234",
// }, allocator);
// try vlc.spawn();
return .{
.allocator = allocator,
.http = std.http.Client{ .allocator = allocator },
.base_uri = base_uri,
.authorization = authorization,
};
} }
};
fn getSongInfo(allocator: std.mem.Allocator) !SongInfo { pub fn deinit(self: *VLC) void {
var http = std.http.Client{ .allocator = allocator }; self.http.deinit();
defer http.deinit(); self.allocator.free(self.authorization);
// try vlc.kill();
}
const userpass = try std.fmt.allocPrint(allocator, ":{s}", .{"1234"}); fn request(self: *VLC, uri: std.Uri) !Response {
defer allocator.free(userpass); var combined_uri = self.base_uri;
combined_uri.path = uri.path;
combined_uri.query = uri.query;
combined_uri.fragment = uri.fragment;
const base64_userpass = try allocator.alloc(u8, base64_encoder.calcSize(userpass.len)); var response = std.ArrayList(u8).init(self.allocator);
defer allocator.free(base64_userpass); defer response.deinit();
const final_userpass = try std.fmt.allocPrint(allocator, "Basic {s}", .{base64_encoder.encode(base64_userpass, userpass)}); const result = try self.http.fetch(.{
defer allocator.free(final_userpass); .location = .{ .uri = combined_uri },
.headers = .{ .authorization = .{ .override = self.authorization } },
.response_storage = .{ .dynamic = &response },
});
var response = std.ArrayList(u8).init(allocator); // std.debug.print("{any}\n{s}\n", .{ result, response.items });
defer response.deinit();
const result = try http.fetch(.{ if (result.status != .ok) return error.HttpRequestFailed;
.location = .{ .url = "http://localhost:8080/requests/status.xml" },
.headers = .{ .authorization = .{ .override = final_userpass } },
.response_storage = .{ .dynamic = &response },
});
std.debug.print("{any}\n{s}\n", .{ result, response.items }); const buffer = try response.toOwnedSlice();
return .{
.buffer = buffer,
.document = try xml.parse(self.allocator, buffer),
};
}
const document = try xml.parse(allocator, response.items); const Response = struct {
defer document.deinit(); buffer: []u8,
document: xml.Document,
var title: ?[]const u8 = null; pub fn deinit(self: *Response, allocator: std.mem.Allocator) void {
var album: ?[]const u8 = null; self.document.deinit();
var artist: ?[]const u8 = null; allocator.free(self.buffer);
}
};
if (document.root.findChildByTag("information")) |information| { pub const SongInfo = struct {
var categories_it = information.findChildrenByTag("category"); allocator: std.mem.Allocator,
while (categories_it.next()) |category| { title: ?[]const u8,
if (std.mem.eql(u8, category.getAttribute("name").?, "meta")) { album: ?[]const u8,
var info_it = category.findChildrenByTag("info"); artist: ?[]const u8,
while (info_it.next()) |info| {
const info_name = info.getAttribute("name").?; // TODO move allocator into struct
if (std.mem.eql(u8, info_name, "title")) pub fn deinit(self: *SongInfo) void {
title = try processHtmlString(allocator, info.children[0].char_data) if (self.title) |b| self.allocator.free(b);
else if (std.mem.eql(u8, info_name, "album")) if (self.album) |b| self.allocator.free(b);
album = try processHtmlString(allocator, info.children[0].char_data) if (self.artist) |b| self.allocator.free(b);
else if (std.mem.eql(u8, info_name, "artist")) }
artist = try processHtmlString(allocator, info.children[0].char_data); };
pub fn getSongInfo(self: *VLC) !SongInfo {
var response = try self.request(.{ .scheme = "http", .path = .{ .percent_encoded = "/requests/status.xml" } });
defer response.deinit(self.allocator);
// std.debug.print("{s}\n", .{response.buffer});
var title: ?[]const u8 = null;
var album: ?[]const u8 = null;
var artist: ?[]const u8 = null;
if (response.document.root.findChildByTag("information")) |information| {
var categories_it = information.findChildrenByTag("category");
while (categories_it.next()) |category| {
if (std.mem.eql(u8, category.getAttribute("name").?, "meta")) {
var info_it = category.findChildrenByTag("info");
while (info_it.next()) |info| {
const info_name = info.getAttribute("name").?;
if (std.mem.eql(u8, info_name, "title"))
title = try processHtmlString(self, info.children[0].char_data)
else if (std.mem.eql(u8, info_name, "album"))
album = try processHtmlString(self, info.children[0].char_data)
else if (std.mem.eql(u8, info_name, "artist"))
artist = try processHtmlString(self, info.children[0].char_data);
}
} }
} }
} }
return .{
.allocator = self.allocator,
.title = title,
.album = album,
.artist = artist,
};
} }
return .{ // TODO clean up
.title = title, fn processHtmlString(self: *VLC, string: []const u8) ![]const u8 {
.album = album, var new: []u8 = try self.allocator.dupe(u8, string);
.artist = artist, errdefer self.allocator.free(new);
}; while (true) {
} if (std.mem.indexOf(u8, new, "&#")) |amp| {
if (std.mem.indexOfScalarPos(u8, new, amp, ';')) |semi| {
fn processHtmlString(allocator: std.mem.Allocator, string: []const u8) ![]const u8 { const int = try std.fmt.parseInt(u8, new[amp + 2 .. semi], 10);
var new: []u8 = try allocator.dupe(u8, string); const nnew = try self.allocator.alloc(u8, std.mem.replacementSize(u8, new, new[amp .. semi + 1], &[1]u8{int}));
errdefer allocator.free(new); _ = std.mem.replace(u8, new, new[amp .. semi + 1], &[1]u8{int}, nnew);
while (true) { self.allocator.free(new);
if (std.mem.indexOf(u8, new, "&#")) |amp| { new = nnew;
if (std.mem.indexOfScalarPos(u8, new, amp, ';')) |semi| { }
const int = try std.fmt.parseInt(u8, new[amp + 2 .. semi], 10); } else break;
const nnew = try allocator.alloc(u8, std.mem.replacementSize(u8, new, new[amp .. semi + 1], &[1]u8{int})); }
_ = std.mem.replace(u8, new, new[amp .. semi + 1], &[1]u8{int}, nnew); // std.debug.print("{s}\n", .{new});
allocator.free(new); return new;
new = nnew;
}
} else break;
} }
std.debug.print("{s}\n", .{new});
return new; const xml = @import("./xml.zig");
} };
fn updateTime(tz: std.Tz) !void { fn updateTime(tz: std.Tz) !void {
const original_timestamp = std.time.timestamp(); const original_timestamp = std.time.timestamp();