zbt

Unnamed repository; edit this file 'description' to name the repository.
Log | Files | Refs | README

commit e3c5be0a33e66b124ff261479cffa9f95ce53dc0
parent e51c15cc478b6c89e9100017eebfdf88aff95cca
Author: Martin Ashby <martin@ashbysoft.com>
Date:   Sat, 11 Nov 2023 22:17:31 +0000

Make request to tracker, get and parse response to find peers list

Diffstat:
Msrc/main.zig | 97++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++---
Msrc/metainfo.zig | 11++++++++++-
2 files changed, 104 insertions(+), 4 deletions(-)

diff --git a/src/main.zig b/src/main.zig @@ -1,10 +1,101 @@ const std = @import("std"); +const MetaInfo = @import("metainfo.zig"); +const bencode = @import("bencode.zig"); pub fn main() !void { - std.log.info("Hello, World", .{}); + var gpa = std.heap.GeneralPurposeAllocator(.{}){}; + defer _ = gpa.deinit(); + const a = gpa.allocator(); + + // open & parse the torrent file + const f = try std.fs.cwd().openFile("src/sample.torrent", .{}); + defer f.close(); + var fr = f.reader(); + var mib = try bencode.bdecode(a, fr); + defer mib.deinit(a); + var mi = try MetaInfo.parse(a, mib); + defer mi.deinit(a); + + // call the tracker... + var c = std.http.Client{ + .allocator = a, + }; + defer c.deinit(); + + var q = std.StringHashMap([]const u8).init(a); + defer q.deinit(); + var info_hash = try mi.info.hash(a); + var buf_left = [_]u8{0} ** 1024; + try q.put("info_hash", &info_hash); + try q.put("peer_id", "00112233445566778899"); + try q.put("port", "6881"); + try q.put("uploaded", "0"); + try q.put("downloaded", "0"); + try q.put("left", try std.fmt.bufPrint(&buf_left, "{}", .{mi.info.files[0].length})); + try q.put("compact", "1"); + var qs = try toqs(a, q); + defer a.free(qs); + var url = try std.fmt.allocPrint(a, "{s}?{s}", .{ mi.announce, qs }); + defer a.free(url); + var res = try c.fetch(a, .{ .location = .{ .url = url } }); + defer res.deinit(); + if (res.status == .ok) { + var trb = try bencode.bdecodeBuf(a, res.body.?); + defer trb.deinit(a); + var tr = try TrackerResp.parse(a, trb); + defer tr.deinit(a); + var w = std.io.getStdOut().writer(); + for (tr.peers) |peer| { + try std.fmt.format(w, "peer: {}\n", .{peer}); + } + } +} + +fn toqs(a: std.mem.Allocator, hm: std.StringHashMap([]const u8)) ![]const u8 { + var al = std.ArrayList(u8).init(a); + defer al.deinit(); + var w = al.writer(); + var it = hm.iterator(); + var first = true; + while (it.next()) |entry| { + if (!first) try w.writeByte('&'); + try std.Uri.writeEscapedQuery(w, entry.key_ptr.*); + try w.writeByte('='); + try std.Uri.writeEscapedQuery(w, entry.value_ptr.*); + first = false; + } + return try al.toOwnedSlice(); } +const TrackerResp = struct { + // interval: u64, + peers: []std.net.Address, + + pub fn parse(a: std.mem.Allocator, b: bencode.BValue) !TrackerResp { + var ipl = std.ArrayList(std.net.Address).init(a); + defer ipl.deinit(); + var d = b.asDict() catch return error.Malformatted; + var pb = d.get("peers") orelse return error.Malformatted; + var ps = pb.asString() catch return error.Malformatted; + if ((ps.len % 6) != 0) return error.Malformatted; + for (0..ps.len / 6) |ix| { + const start = ix * 6; + const port = std.mem.readInt(u16, ps[start + 4 .. start + 6][0..2], .big); + var ip = [_]u8{0} ** 4; + @memcpy(&ip, ps[start .. start + 4]); + try ipl.append(std.net.Address.initIp4(ip, port)); + } + return .{ + .peers = try ipl.toOwnedSlice(), + }; + } + + pub fn deinit(self: *TrackerResp, a: std.mem.Allocator) void { + a.free(self.peers); + } +}; + test { - _ = @import("bencode.zig"); - _ = @import("metainfo.zig"); + _ = bencode; + _ = MetaInfo; } diff --git a/src/metainfo.zig b/src/metainfo.zig @@ -63,9 +63,13 @@ pub const Info = struct { const pri = pr.asInt(u1) catch return error.Malformatted; priv = pri == 1; } + // Validation.... + const ps = pp.asString() catch return error.Malformatted; + if (ps.len % 20 != 0) return error.Malformatted; + return .{ .piece_length = pl.asInt(u64) catch return error.Malformatted, - .pieces = pp.asString() catch return error.Malformatted, + .pieces = ps, .files = try files.toOwnedSlice(), .private = priv, }; @@ -112,6 +116,11 @@ pub const Info = struct { try b.bencode(w); return sha1.finalResult(); } + pub fn pieceHash(self: Info, ix: usize) ?[]const u8 { + const start = 20 * ix; + if (start >= self.pieces.len) return null; + return self.pieces[start .. start + 20]; + } }; info: Info,