shelving this version for a bit

This commit is contained in:
Jeeves 2025-02-18 16:06:08 -07:00
parent b419eeb1c4
commit 9b84d6f651
3 changed files with 100 additions and 68 deletions

View file

@ -14,12 +14,6 @@ const std = @import("std");
// //
// A - means immediately, a + means eventually. // A - means immediately, a + means eventually.
var global_connection: ?std.net.Server.Connection = null;
var global_websocket: ?*std.http.WebSocket = null;
var global_send_buf: ?[]u8 = null;
var global_recv_buf: ?[]align(4) u8 = null;
pub fn main() !void { pub fn main() !void {
var gpa = std.heap.GeneralPurposeAllocator(.{}){}; var gpa = std.heap.GeneralPurposeAllocator(.{}){};
defer _ = gpa.deinit(); defer _ = gpa.deinit();
@ -28,30 +22,33 @@ pub fn main() !void {
var vlc = try VLC.init(allocator, try std.Uri.parse("http://localhost:8080")); var vlc = try VLC.init(allocator, try std.Uri.parse("http://localhost:8080"));
defer vlc.deinit(); defer vlc.deinit();
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();
var tz = try std.tz.Tz.parse(allocator, tz_file.reader()); // var tz = try std.tz.Tz.parse(allocator, tz_file.reader());
defer tz.deinit(); // defer tz.deinit();
var server = try std.net.Address.initIp4(.{ 127, 0, 0, 1 }, 8009).listen(.{}); var server = try std.net.Address.initIp4(.{ 127, 0, 0, 1 }, 8009).listen(.{});
defer server.deinit(); defer server.deinit();
// while (true) : (std.time.sleep(1_000_000_000)) { // sleep 500ms var clients = std.ArrayList(Client).init(allocator);
// try updateTime(tz); defer {
// try updateStatus(allocator, &vlc); for (0..clients.items.len) |i| clients.items[i].deinit(allocator);
// // scroll += 1; clients.deinit();
// } }
defer if (global_connection) |conn| conn.stream.close(); // var delay: u8 = 0;
defer if (global_send_buf) |buf| allocator.free(buf);
defer if (global_recv_buf) |buf| allocator.free(buf);
var delay: u8 = 0;
while (true) { while (true) {
var pollfds = [_]std.posix.pollfd{.{ .fd = server.stream.handle, .events = std.posix.POLL.IN, .revents = 0 }}; var pollfds = std.ArrayList(std.posix.pollfd).init(allocator);
const num = try std.posix.poll(&pollfds, 100); defer pollfds.deinit();
std.debug.print("num pollfds {d}\nrevents {d}\n", .{ num, pollfds[0].revents }); try pollfds.append(.{ .fd = server.stream.handle, .events = std.posix.POLL.IN, .revents = 0 });
if (pollfds[0].revents != 0) { for (clients.items, 0..) |client, i| {
std.debug.print("appending client {d} to poll()\n", .{i});
try pollfds.append(.{ .fd = client.connection.stream.handle, .events = std.posix.POLL.IN, .revents = 0 });
}
const num = try std.posix.poll(pollfds.items, 100);
// std.debug.print("num pollfds {d}\n", .{num});
_ = num;
if (pollfds.items[0].revents != 0) {
const read_buf = try allocator.alloc(u8, 1024 * 32); const read_buf = try allocator.alloc(u8, 1024 * 32);
defer allocator.free(read_buf); defer allocator.free(read_buf);
@ -64,24 +61,26 @@ pub fn main() !void {
var websocket: std.http.WebSocket = undefined; var websocket: std.http.WebSocket = undefined;
global_send_buf = try allocator.alloc(u8, 1024 * 32); const send_buf = try allocator.alloc(u8, 1024 * 32);
global_recv_buf = try allocator.alignedAlloc(u8, 4, 1024 * 32); errdefer allocator.free(send_buf);
const recv_buf = try allocator.alignedAlloc(u8, 4, 1024 * 32);
errdefer allocator.free(recv_buf);
const is_websocket = try websocket.init(&request, global_send_buf.?, global_recv_buf.?); const is_websocket = try websocket.init(&request, send_buf, recv_buf);
if (is_websocket) { if (is_websocket) {
// if websocket, we clean up at the end of main(), not the end of the block
try websocket.response.flush(); try websocket.response.flush();
std.debug.print("is a websocket now\n", .{}); std.debug.print("is a websocket now\n", .{});
global_connection = connection;
global_websocket = &websocket;
// const msg = try websocket.readSmallMessage();
// std.debug.print("{any}\n{s}\n", .{ msg, msg.data });
// try websocket.writeMessage("dude i cant believe you just ate that", .text); const client = Client{
// // try websocket.writeMessage("", .ping); .connection = connection,
// std.time.sleep(5_000_000_000); .websocket = websocket,
// try websocket.writeMessage("dude why did i just eat that now", .text); .send_buf = send_buf,
// std.debug.print("sent second one\n", .{}); .recv_buf = recv_buf,
};
try clients.append(client);
} else { } else {
// if normal HTTP, clean up at the end of the block
defer connection.stream.close(); defer connection.stream.close();
switch (request.head.method) { switch (request.head.method) {
.GET, .HEAD => { .GET, .HEAD => {
@ -104,25 +103,46 @@ pub fn main() !void {
} }
} }
delay += 1; const status_string = try getStatusString(allocator, &vlc);
if (delay == 10) { defer allocator.free(status_string);
delay = 0; for (1..pollfds.items.len) |i| {
try updateStatus(allocator, &vlc); const pollfd = pollfds.items[i];
var client = &clients.items[i - 1];
std.debug.print("client idx {d} for status update\nrevents {d}\n", .{ i - 1, pollfd.revents });
// try std.http.WebSocket.writeMessage(&client.websocket, status_string, .text);
if (pollfd.revents & std.posix.POLL.IN != 0) {
const data = std.http.WebSocket.readSmallMessage(&client.websocket) catch |e| switch (e) {
error.EndOfStream => continue,
error.ConnectionClose => continue,
else => return e,
};
std.debug.print("recieved opcode: {any} with data {s}\n", .{ data.opcode, data.data });
} }
} }
std.Thread.sleep(1_000_000_000);
}
} }
const Client = struct {
connection: std.net.Server.Connection,
websocket: std.http.WebSocket,
send_buf: []u8,
recv_buf: []align(4) u8,
pub fn deinit(self: *Client, allocator: std.mem.Allocator) void {
allocator.free(self.send_buf);
allocator.free(self.recv_buf);
self.connection.stream.close();
}
};
const base64_encoder = std.base64.standard.Encoder; const base64_encoder = std.base64.standard.Encoder;
// TODO make the URL something short like jeevio.xyz/streamboy // 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; fn getStatusString(allocator: std.mem.Allocator, vlc: *VLC) ![]u8 {
fn updateStatus(allocator: std.mem.Allocator, vlc: *VLC) !void {
// var stream_info_file = try std.fs.createFileAbsolute("/tmp/streaminfo", .{ .truncate = true });
// defer stream_info_file.close();
if (global_websocket) |ws| {
var song_info = try vlc.getSongInfo(); var song_info = try vlc.getSongInfo();
defer song_info.deinit(); defer song_info.deinit();
@ -131,16 +151,9 @@ fn updateStatus(allocator: std.mem.Allocator, vlc: *VLC) !void {
song_info.title orelse "Unknown Title", song_info.title orelse "Unknown Title",
song_info.artist orelse "Unknown Artist", song_info.artist orelse "Unknown Artist",
}); });
defer allocator.free(string); errdefer allocator.free(string);
try ws.writeMessage(string, .text); return string;
}
// if (scroll > string.len) scroll = 0;
// if (scroll == 0) try stream_info_file.writeAll(string) else {
// for (string[scroll..]) |char| try stream_info_file.writer().writeByte(char);
// for (string[0..scroll]) |char| try stream_info_file.writer().writeByte(char);
// }
} }
pub const VLC = struct { pub const VLC = struct {
@ -159,6 +172,7 @@ pub const VLC = struct {
const authorization = try std.fmt.allocPrint(allocator, "Basic {s}", .{base64_encoder.encode(base64_userpass, userpass)}); const authorization = try std.fmt.allocPrint(allocator, "Basic {s}", .{base64_encoder.encode(base64_userpass, userpass)});
errdefer allocator.free(authorization); errdefer allocator.free(authorization);
// TODO actually launch VLC
// var vlc = std.process.Child.init(&[_][]const u8{ // var vlc = std.process.Child.init(&[_][]const u8{
// "vlc", // "vlc",
// "--intf", // "--intf",
@ -226,7 +240,6 @@ pub const VLC = struct {
album: ?[]const u8, album: ?[]const u8,
artist: ?[]const u8, artist: ?[]const u8,
// TODO move allocator into struct
pub fn deinit(self: *SongInfo) void { pub fn deinit(self: *SongInfo) void {
if (self.title) |b| self.allocator.free(b); if (self.title) |b| self.allocator.free(b);
if (self.album) |b| self.allocator.free(b); if (self.album) |b| self.allocator.free(b);

View file

@ -2,11 +2,18 @@ const statusLineElement = document.getElementById("status-line");
const socket = new WebSocket("ws://localhost:8009"); const socket = new WebSocket("ws://localhost:8009");
// socket.addEventListener("open", (event) => {
// socket.send("Hello Server!");
// });
socket.addEventListener("message", (event) => { socket.addEventListener("message", (event) => {
console.log("Message from server ", event.data); console.log(event);
statusLineElement.textContent = event.data; statusLineElement.textContent = event.data;
}); });
// const status = {
// blocks: [
// "Something Fun For Everyone With Streamboy!!!!!!!! git.jeevio.xyz/jeeves/streamboy",
// "♪ Bonhomme de Neige (EarthBound) - Ridley Snipes feat. Earth Kid",
// ],
// };
// setInterval(() => {
// // statusText = "Something Fun For Everyone With Streamboy!!!!!!!! git.jeevio.xyz/jeeves/streamboy | ♪ Bonhomme de Neige (EarthBound) - Ridley Snipes feat. Earth Kid | ";
// }, 5000);

View file

@ -13,4 +13,16 @@ body {
#status-line { #status-line {
font-size: 16px; font-size: 16px;
margin: 0px; margin: 0px;
white-space: nowrap;
animation: marquee 5s linear infinite;
max-width: none;
}
@keyframes marquee {
from {
left: -100%;
}
to {
left: 100%;
}
} }