summaryrefslogtreecommitdiff
path: root/src/parser.zig
blob: ba1655af5c4c1e974d14f49e37ab1b0ea67b652c (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
const std = @import("std");

pub const ParsedCommand = struct {
    name: []const u8,
    args: ?[]const u8,
};

pub fn parseCommand(input: []const u8) ParsedCommand {
    const space_pos = std.mem.indexOfScalar(u8, input, ' ');
    if (space_pos) |pos| {
        return .{ .name = input[0..pos], .args = input[pos + 1 ..] };
    }
    return .{ .name = input, .args = null };
}

pub fn parseArgs(allocator: std.mem.Allocator, cmd_name: []const u8, args_str: ?[]const u8) ![]const []const u8 {
    var args_list = std.ArrayList([]const u8){};
    try args_list.ensureTotalCapacity(allocator, 16);
    errdefer args_list.deinit(allocator);

    try args_list.append(allocator, cmd_name);

    if (args_str) |args| {
        var i: usize = 0;
        while (i < args.len) {
            if (args[i] == '\'' or args[i] == '"') {
                const quote = args[i];
                i += 1;
                const start = i;
                while (i < args.len and args[i] != quote) : (i += 1) {}
                if (i <= args.len) {
                    try args_list.append(allocator, args[start..i]);
                    i += 1;
                }
            } else if (args[i] != ' ') {
                const start = i;
                while (i < args.len and args[i] != ' ' and args[i] != '\'' and args[i] != '"') : (i += 1) {}
                try args_list.append(allocator, args[start..i]);
            } else {
                i += 1;
            }
        }
    }

    return args_list.toOwnedSlice(allocator);
}