summaryrefslogtreecommitdiff
path: root/backend/src/Authentication/Tokens.zig
blob: c713635127451d86449dd5c399771b37d736223d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
const std = @import("std");
const redis = @import("../Redis/Connection.zig");
const userModel = @import("../Models/User.zig");

const topicName = "cp2020_auth";
const token_ttl: u16 = 43_200; // 12 hours

var prng: std.Random.DefaultPrng= undefined;
var rnd: std.Random = undefined;

pub const errors = error {
    NotFound,
};

pub fn Init() !void {
    // Random enough
    var seed: u64 = undefined;
    try std.posix.getrandom(std.mem.asBytes(&seed));
    prng = std.Random.DefaultPrng.init(seed);
    rnd = prng.random();
}

/// Generates a new token in the stack. Expects [64]u8 as an input 
fn generateSessionToken(buf: []u8) []u8 {
    for (0..64) |i| {
        buf[i] = std.Random.intRangeAtMost(rnd, u8, 'A', 'z');
        if (buf[i] == '\\') {
            buf[i] = '@';
        }
    }
    return buf;
}

pub fn GenerateNewSession(allocator: std.mem.Allocator, user: userModel.User) ![]u8 {
    const token: []u8 = try allocator.alloc(u8, 64);
    const newToken = generateSessionToken(token);

    // Redis cache 
    var string: std.io.Writer.Allocating = .init(allocator);
    defer string.deinit();
    try string.writer.print("{f}", .{ std.json.fmt(user, .{}) });
    try redis.WriteToTopic(topicName, redis.Message {
        .Key = token,
        .Value = string.written(),
        .SecondsToLive = token_ttl,
    });

    return newToken;
}

pub fn GetUserFromToken(allocator: std.mem.Allocator, token: []const u8) !?std.json.Parsed(userModel.User) {
    const saved = try redis.ReadFromTopic(allocator, topicName, token) orelse {
        // std.debug.print("The token was not found \n", .{});
        return errors.NotFound;
    };

    const parsed = try std.json.parseFromSlice(
        userModel.User,
        allocator,
        saved, 
        .{ 
            .ignore_unknown_fields = true,
        });
    return parsed;
}