clean up VLC-related code
This commit is contained in:
parent
a7ca2f9377
commit
6ce0d30b4c
1 changed files with 137 additions and 84 deletions
221
src/main.zig
221
src/main.zig
|
@ -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();
|
||||||
|
|
Loading…
Add table
Reference in a new issue