finish
This commit is contained in:
parent
607bf5021e
commit
8fb7891fe1
200
src/base64.zig
200
src/base64.zig
@ -5,105 +5,103 @@ pub const err = error{
|
|||||||
IndexNotFound,
|
IndexNotFound,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub const Base64 = struct {
|
fn char_at(index: u8) u8 {
|
||||||
table: *const [64]u8,
|
const table: *const [64]u8 = comptime "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
||||||
|
return table[index];
|
||||||
|
}
|
||||||
|
|
||||||
pub const init: Base64 = .{
|
pub fn encode(allocator: std.mem.Allocator, input: []const u8) ![]u8 {
|
||||||
.table = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",
|
if (input.len == 0) {
|
||||||
};
|
return "";
|
||||||
|
|
||||||
fn char_at(self: Base64, index: u8) u8 {
|
|
||||||
return self.table[index];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn encode(self: Base64, allocator: std.mem.Allocator, input: []const u8) ![]u8 {
|
const out_sz = try calc_encode_length(input);
|
||||||
if (input.len == 0) {
|
var out = try allocator.alloc(u8, out_sz);
|
||||||
return "";
|
var buf = [3]u8{0, 0, 0};
|
||||||
|
var count: u16 = 0;
|
||||||
|
var outc: u16 = 0;
|
||||||
|
|
||||||
|
for (input) |b| {
|
||||||
|
buf[count] = b;
|
||||||
|
count += 1;
|
||||||
|
if (count == 3) {
|
||||||
|
out[outc] = char_at(buf[0] >> 2);
|
||||||
|
out[outc + 1] = char_at(((buf[0] & 0x03) << 4) + (buf[1] >> 4));
|
||||||
|
out[outc + 2] = char_at(((buf[1] & 0x0F) << 2) + (buf[2] >> 6));
|
||||||
|
out[outc + 3] = char_at(buf[2] & 0x3F);
|
||||||
|
count = 0;
|
||||||
|
outc += 4;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const encode_length = try calc_encode_length(input);
|
if (count == 2) {
|
||||||
var out = try allocator.alloc(u8, encode_length);
|
out[outc] = char_at(buf[0] >> 2);
|
||||||
var buf = [3]u8{0, 0, 0};
|
out[outc + 1] = char_at((buf[0] & 0x03) << 4) + (buf[1] >> 4);
|
||||||
var count: u16 = 0;
|
out[outc + 2] = char_at((buf[1] & 0x0F) << 2);
|
||||||
var outc: u16 = 0;
|
out[outc + 3] = '=';
|
||||||
|
} else if (count == 1) {
|
||||||
|
out[outc] = char_at(buf[0] >> 2);
|
||||||
|
out[outc + 1] = char_at((buf[0] & 0x03) << 4);
|
||||||
|
out[outc + 2] = '=';
|
||||||
|
out[outc + 3] = '=';
|
||||||
|
}
|
||||||
|
|
||||||
for (input) |b| {
|
return out;
|
||||||
buf[count] = b;
|
}
|
||||||
count += 1;
|
|
||||||
if (count == 3) {
|
fn decode_index(char: u8) err!u8 {
|
||||||
out[outc] = self.char_at(buf[0] >> 2);
|
if (char == '=') {
|
||||||
out[outc + 1] = self.char_at(((buf[0] & 0x03) << 4) + (buf[1] >> 4));
|
return 64;
|
||||||
out[outc + 2] = self.char_at(((buf[1] & 0x0F) << 2) + (buf[2] >> 6));
|
}
|
||||||
out[outc + 3] = self.char_at(buf[2] & 0x3F);
|
|
||||||
count = 0;
|
if (char >= 'A' and char <= 'Z') {
|
||||||
outc += 4;
|
return char - 'A';
|
||||||
|
} else if (char >= 'a' and char <= 'z') {
|
||||||
|
return char - 'a' + 26;
|
||||||
|
} else if (char >= '0' and char <= '9') {
|
||||||
|
return char - '0' + 52;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (char) {
|
||||||
|
'+' => { return 62; },
|
||||||
|
'/' => { return 63; },
|
||||||
|
'=' => { return 64; },
|
||||||
|
else => { return err.IndexNotFound; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn decode(allocator: std.mem.Allocator, input: []const u8) ![]u8 {
|
||||||
|
if (input.len == 0) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
const out_sz = try calc_decode_length(input);
|
||||||
|
var out = try allocator.alloc(u8, out_sz);
|
||||||
|
var count: u8 = 0;
|
||||||
|
var iout: u64 = 0;
|
||||||
|
var buf = [4]u8{ 0, 0, 0, 0 };
|
||||||
|
var cutoff :u8 = 0;
|
||||||
|
|
||||||
|
for (0..input.len) |i| {
|
||||||
|
buf[count] = try decode_index(input[i]);
|
||||||
|
count += 1;
|
||||||
|
if (count == 4) {
|
||||||
|
out[iout] = (buf[0] << 2) + (buf[1] >> 4);
|
||||||
|
if (buf[2] != 64) {
|
||||||
|
out[iout + 1] = (buf[1] << 4) + (buf[2] >> 2);
|
||||||
|
cutoff = 2;
|
||||||
}
|
}
|
||||||
}
|
if (buf[3] != 64) {
|
||||||
|
out[iout + 2] = (buf[2] << 6) + buf[3];
|
||||||
if (count == 2) {
|
cutoff = 1;
|
||||||
out[outc] = self.char_at(buf[0] >> 2);
|
|
||||||
out[outc + 1] = self.char_at((buf[0] & 0x03) << 4) + (buf[1] >> 4);
|
|
||||||
out[outc + 2] = self.char_at((buf[1] & 0x0F) << 2);
|
|
||||||
out[outc + 3] = '=';
|
|
||||||
} else if (count == 1) {
|
|
||||||
out[outc] = self.char_at(buf[0] >> 2);
|
|
||||||
out[outc + 1] = self.char_at((buf[0] & 0x03) << 4);
|
|
||||||
out[outc + 2] = '=';
|
|
||||||
out[outc + 3] = '=';
|
|
||||||
}
|
|
||||||
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
|
|
||||||
fn decode_index(self: Base64, char: u8) err!u8 {
|
|
||||||
if (char == '=') {
|
|
||||||
return 64;
|
|
||||||
}
|
|
||||||
if (char >= 'A' and char <= 'Z') {
|
|
||||||
return char - 'A';
|
|
||||||
} else if (char >= 'a' and char <= 'z') {
|
|
||||||
return char - 'a' + 26;
|
|
||||||
} else {
|
|
||||||
for (52..64) |i| {
|
|
||||||
const idx: u8 = @intCast(i);
|
|
||||||
if (self.char_at(idx) == char) {
|
|
||||||
return idx;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
iout += 3;
|
||||||
|
count = 0;
|
||||||
}
|
}
|
||||||
return err.IndexNotFound;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn decode(self: Base64, allocator: std.mem.Allocator, input: []const u8) ![]u8 {
|
return out[0..out.len - cutoff + 1];
|
||||||
if (input.len == 0) {
|
}
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
const output_sz = try calc_decode_length(input);
|
|
||||||
var output = try allocator.alloc(u8, output_sz);
|
|
||||||
var count: u8 = 0;
|
|
||||||
var iout: u64 = 0;
|
|
||||||
var buf = [4]u8{ 0, 0, 0, 0 };
|
|
||||||
|
|
||||||
for (0..input.len) |i| {
|
|
||||||
buf[count] = try self.decode_index(input[i]);
|
|
||||||
count += 1;
|
|
||||||
if (count == 4) {
|
|
||||||
output[iout] = (buf[0] << 2) + (buf[1] >> 4);
|
|
||||||
if (buf[2] != 64) {
|
|
||||||
output[iout + 1] = (buf[1] << 4) + (buf[2] >> 2);
|
|
||||||
}
|
|
||||||
if (buf[3] != 64) {
|
|
||||||
output[iout + 2] = (buf[2] << 6) + buf[3];
|
|
||||||
}
|
|
||||||
iout += 3;
|
|
||||||
count = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return output;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
fn calc_encode_length(input: []const u8) !usize {
|
fn calc_encode_length(input: []const u8) !usize {
|
||||||
if (input.len < 3) {
|
if (input.len < 3) {
|
||||||
@ -122,57 +120,51 @@ fn calc_decode_length(input: []const u8) !usize {
|
|||||||
}
|
}
|
||||||
|
|
||||||
test "encode hello" {
|
test "encode hello" {
|
||||||
var b = Base64.init;
|
|
||||||
var gpa = std.heap.GeneralPurposeAllocator(.{}).init;
|
var gpa = std.heap.GeneralPurposeAllocator(.{}).init;
|
||||||
const allocator = gpa.allocator();
|
const allocator = gpa.allocator();
|
||||||
|
|
||||||
const encoded = try b.encode(allocator, "hello");
|
const encoded = try encode(allocator, "hello");
|
||||||
try std.testing.expect(std.mem.eql(u8, encoded, "aGVsbG8="));
|
try std.testing.expect(std.mem.eql(u8, encoded, "aGVsbG8="));
|
||||||
}
|
}
|
||||||
|
|
||||||
test "encode long" {
|
test "encode long" {
|
||||||
var b = Base64.init;
|
|
||||||
var gpa = std.heap.GeneralPurposeAllocator(.{}).init;
|
var gpa = std.heap.GeneralPurposeAllocator(.{}).init;
|
||||||
const allocator = gpa.allocator();
|
const allocator = gpa.allocator();
|
||||||
|
|
||||||
const encoded = try b.encode(allocator, "Hey, it's me. I'm the problem. It's me");
|
const encoded = try encode(allocator, "Hey, it's me. I'm the problem. It's me");
|
||||||
try std.testing.expect(std.mem.eql(u8, encoded, "SGV5LCBpdCdzIG1lLiBJJ20gdGhlIHByb2JsZW0uIEl0J3MgbWU="));
|
try std.testing.expect(std.mem.eql(u8, encoded, "SGV5LCBpdCdzIG1lLiBJJ20gdGhlIHByb2JsZW0uIEl0J3MgbWU="));
|
||||||
}
|
}
|
||||||
|
|
||||||
test "decode hello" {
|
test "decode hello" {
|
||||||
var b = Base64.init;
|
|
||||||
var gpa = std.heap.GeneralPurposeAllocator(.{}).init;
|
var gpa = std.heap.GeneralPurposeAllocator(.{}).init;
|
||||||
const allocator = gpa.allocator();
|
const allocator = gpa.allocator();
|
||||||
|
|
||||||
const encoded = try b.decode(allocator, "aGVsbG8=");
|
const encoded = try decode(allocator, "aGVsbG8=");
|
||||||
try stdout.print("{s}\n", .{encoded});
|
try stdout.print("{s}\n", .{encoded});
|
||||||
try std.testing.expect(std.mem.eql(u8, encoded, "hello"));
|
try std.testing.expect(std.mem.eql(u8, encoded, "hello"));
|
||||||
}
|
}
|
||||||
|
|
||||||
test "decode long" {
|
test "decode long" {
|
||||||
var b = Base64.init;
|
|
||||||
var gpa = std.heap.GeneralPurposeAllocator(.{}).init;
|
var gpa = std.heap.GeneralPurposeAllocator(.{}).init;
|
||||||
const allocator = gpa.allocator();
|
const allocator = gpa.allocator();
|
||||||
|
|
||||||
const encoded = try b.decode(allocator, "SGV5LCBpdCdzIG1lLiBJJ20gdGhlIHByb2JsZW0uIEl0J3MgbWU=");
|
const encoded = try decode(allocator, "SGV5LCBpdCdzIG1lLiBJJ20gdGhlIHByb2JsZW0uIEl0J3MgbWU=");
|
||||||
try stdout.print("{s}\n", .{encoded});
|
try stdout.print("{s}\n", .{encoded});
|
||||||
try std.testing.expect(std.mem.eql(u8, encoded, "Hey, it's me. I'm the problem. It's me"));
|
try std.testing.expect(std.mem.eql(u8, encoded, "Hey, it's me. I'm the problem. It's me"));
|
||||||
}
|
}
|
||||||
|
|
||||||
test "decode_index" {
|
test "decode_index" {
|
||||||
var b = Base64.init;
|
try std.testing.expectError(err.IndexNotFound, decode_index('{'));
|
||||||
|
|
||||||
try std.testing.expectError(err.IndexNotFound, b.decode_index('{'));
|
|
||||||
|
|
||||||
var r :u8 = 0;
|
var r :u8 = 0;
|
||||||
r = try b.decode_index('A');
|
r = try decode_index('A');
|
||||||
try std.testing.expect(r == 0);
|
try std.testing.expect(r == 0);
|
||||||
r = try b.decode_index('a');
|
r = try decode_index('a');
|
||||||
try std.testing.expect(r == 26);
|
try std.testing.expect(r == 26);
|
||||||
r = try b.decode_index('0');
|
r = try decode_index('0');
|
||||||
try std.testing.expect(r == 52);
|
try std.testing.expect(r == 52);
|
||||||
r = try b.decode_index('/');
|
r = try decode_index('/');
|
||||||
try std.testing.expect(r == 63);
|
try std.testing.expect(r == 63);
|
||||||
r = try b.decode_index('=');
|
r = try decode_index('=');
|
||||||
try std.testing.expect(r == 64);
|
try std.testing.expect(r == 64);
|
||||||
}
|
}
|
||||||
|
34
src/main.zig
34
src/main.zig
@ -1,7 +1,7 @@
|
|||||||
const std = @import("std");
|
const std = @import("std");
|
||||||
const stdout = std.io.getStdOut().writer();
|
const stdout = std.io.getStdOut().writer();
|
||||||
|
|
||||||
const b64 = @import("base64.zig");
|
const Base64 = @import("base64.zig");
|
||||||
|
|
||||||
const usageText =
|
const usageText =
|
||||||
\\Usage: baze64 {-d/-D} input
|
\\Usage: baze64 {-d/-D} input
|
||||||
@ -16,22 +16,34 @@ pub fn main() !void {
|
|||||||
var args = try std.process.argsWithAllocator(allocator);
|
var args = try std.process.argsWithAllocator(allocator);
|
||||||
defer args.deinit();
|
defer args.deinit();
|
||||||
|
|
||||||
var b = b64.Base64.init;
|
var decodeSwitch = false;
|
||||||
|
var input: []const u8 = undefined;
|
||||||
|
|
||||||
// Skip arg[0]
|
// Skip arg[0]
|
||||||
_ = args.skip();
|
_ = args.skip();
|
||||||
|
|
||||||
var a = args.next();
|
while (true) {
|
||||||
if (a) |arg| {
|
const argVal = args.next();
|
||||||
if (std.mem.eql(u8, arg, "-d") or std.mem.eql(u8, arg, "-D")) {
|
var arg: []const u8 = undefined;
|
||||||
a = args.next();
|
if (argVal) |a| {
|
||||||
const decoded = try b.decode(allocator, arg);
|
arg = a;
|
||||||
try stdout.print("{s}", .{decoded});
|
|
||||||
} else {
|
} else {
|
||||||
const encoded = try b.decode(allocator, arg);
|
break;
|
||||||
try stdout.print("{s}", .{encoded});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (std.mem.eql(u8, arg, "-d") or std.mem.eql(u8, arg, "-D")) {
|
||||||
|
decodeSwitch = true;
|
||||||
|
} else {
|
||||||
|
input = arg;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (decodeSwitch) {
|
||||||
|
const decoded = try Base64.decode(allocator, input);
|
||||||
|
try stdout.print("{s}", .{decoded});
|
||||||
} else {
|
} else {
|
||||||
try stdout.print("{s}", .{usageText});
|
const encoded = try Base64.encode(allocator, input);
|
||||||
|
try stdout.print("{s}", .{encoded});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user