203 lines
7.1 KiB
Zig
203 lines
7.1 KiB
Zig
const std = @import("std");
|
|
const types = @import("types.zig");
|
|
const print = std.debug.print;
|
|
|
|
pub const HELP_TEXT =
|
|
\\Usage: manwhere [OPTIONS] <keyword>
|
|
\\Find man pages containing a keyword.
|
|
\\
|
|
\\OPTIONS:
|
|
\\ -h, --help Show this help message
|
|
\\ -v, --verbose Show timing and extra details
|
|
\\ --paths Show file paths (slower)
|
|
\\ -s, --section NUM Only show results from section NUM (1-9)
|
|
\\ Can be specified multiple times
|
|
\\
|
|
\\EXAMPLES:
|
|
\\ manwhere sleep # find all man pages mentioning "sleep"
|
|
\\ manwhere -s 1 sleep # find only commands (section 1) mentioning "sleep"
|
|
\\ manwhere -s 1 -s 3 sleep # find in sections 1 and 3
|
|
\\ manwhere -v --paths ssl # detailed search with paths and timing
|
|
\\ manwhere --section 3 printf # find only library functions (section 3)
|
|
\\
|
|
;
|
|
|
|
pub fn parseArgs(allocator: std.mem.Allocator, args: [][:0]u8) !types.SearchConfig {
|
|
if (args.len < 2) {
|
|
print("{s}", .{HELP_TEXT});
|
|
return error.NoKeyword;
|
|
}
|
|
|
|
var keyword: []const u8 = "";
|
|
var sections_list = std.ArrayList([]const u8){};
|
|
defer sections_list.deinit(allocator);
|
|
var show_paths = false;
|
|
var verbose = false;
|
|
|
|
var i: usize = 1;
|
|
while (i < args.len) : (i += 1) {
|
|
const arg = args[i];
|
|
|
|
if (std.mem.eql(u8, arg, "-h") or std.mem.eql(u8, arg, "--help")) {
|
|
print("{s}", .{HELP_TEXT});
|
|
return error.HelpRequested;
|
|
} else if (std.mem.eql(u8, arg, "--paths")) {
|
|
show_paths = true;
|
|
} else if (std.mem.eql(u8, arg, "-v") or std.mem.eql(u8, arg, "--verbose")) {
|
|
verbose = true;
|
|
} else if (std.mem.eql(u8, arg, "-s") or std.mem.eql(u8, arg, "--section")) {
|
|
// Next argument should be the section number
|
|
if (i + 1 >= args.len) {
|
|
print("[X] Option {s} requires a section number (1-9)\n", .{arg});
|
|
return error.InvalidOption;
|
|
}
|
|
i += 1;
|
|
const section_arg = args[i];
|
|
|
|
// Validate section number
|
|
if (section_arg.len == 0 or section_arg.len > 4) {
|
|
print("[X] Invalid section '{s}'. Use 1-9 or subsections like 3ssl\n", .{section_arg});
|
|
return error.InvalidSection;
|
|
}
|
|
|
|
// Basic validation - should start with a digit 1-9
|
|
if (section_arg[0] < '1' or section_arg[0] > '9') {
|
|
print("[X] Invalid section '{s}'. Section must be 1-9\n", .{section_arg});
|
|
return error.InvalidSection;
|
|
}
|
|
|
|
// Add section to list
|
|
try sections_list.append(allocator, section_arg);
|
|
} else if (std.mem.startsWith(u8, arg, "-")) {
|
|
print("[X] Unknown option: {s}\nUse -h or --help for usage information\n", .{arg});
|
|
return error.InvalidOption;
|
|
} else if (keyword.len == 0) {
|
|
keyword = arg;
|
|
} else {
|
|
print("[X] Multiple keywords not supported\n", .{});
|
|
return error.MultipleKeywords;
|
|
}
|
|
}
|
|
|
|
if (keyword.len == 0) {
|
|
print("[X] No keyword provided\n{s}", .{HELP_TEXT});
|
|
return error.NoKeyword;
|
|
}
|
|
|
|
// Convert ArrayList to owned slice, or null if empty
|
|
const target_sections = if (sections_list.items.len > 0)
|
|
try sections_list.toOwnedSlice(allocator)
|
|
else
|
|
null;
|
|
|
|
return types.SearchConfig{
|
|
.keyword = keyword,
|
|
.target_sections = target_sections,
|
|
.show_paths = show_paths,
|
|
.verbose = verbose,
|
|
};
|
|
}
|
|
|
|
// ============================================================================
|
|
// TESTS
|
|
// ============================================================================
|
|
|
|
test "parseArgs - basic keyword" {
|
|
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
|
defer _ = gpa.deinit();
|
|
const allocator = gpa.allocator();
|
|
|
|
const args = &[_][:0]u8{ "manwhere", "sleep" };
|
|
const config = try parseArgs(allocator, args);
|
|
|
|
try std.testing.expectEqualStrings("sleep", config.keyword);
|
|
try std.testing.expect(config.target_sections == null);
|
|
try std.testing.expect(!config.show_paths);
|
|
try std.testing.expect(!config.verbose);
|
|
}
|
|
|
|
test "parseArgs - with verbose flag" {
|
|
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
|
defer _ = gpa.deinit();
|
|
const allocator = gpa.allocator();
|
|
|
|
const args = &[_][:0]u8{ "manwhere", "-v", "ls" };
|
|
const config = try parseArgs(allocator, args);
|
|
|
|
try std.testing.expectEqualStrings("ls", config.keyword);
|
|
try std.testing.expect(config.verbose);
|
|
}
|
|
|
|
test "parseArgs - with paths flag" {
|
|
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
|
defer _ = gpa.deinit();
|
|
const allocator = gpa.allocator();
|
|
|
|
const args = &[_][:0]u8{ "manwhere", "--paths", "printf" };
|
|
const config = try parseArgs(allocator, args);
|
|
|
|
try std.testing.expectEqualStrings("printf", config.keyword);
|
|
try std.testing.expect(config.show_paths);
|
|
}
|
|
|
|
test "parseArgs - with section flag" {
|
|
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
|
defer _ = gpa.deinit();
|
|
const allocator = gpa.allocator();
|
|
|
|
const args = &[_][:0]u8{ "manwhere", "-s", "1", "ls" };
|
|
const config = try parseArgs(allocator, args);
|
|
|
|
try std.testing.expectEqualStrings("ls", config.keyword);
|
|
try std.testing.expect(config.target_sections != null);
|
|
try std.testing.expectEqual(@as(usize, 1), config.target_sections.?.len);
|
|
try std.testing.expectEqualStrings("1", config.target_sections.?[0]);
|
|
}
|
|
|
|
test "parseArgs - with multiple sections" {
|
|
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
|
defer _ = gpa.deinit();
|
|
const allocator = gpa.allocator();
|
|
|
|
const args = &[_][:0]u8{ "manwhere", "-s", "1", "-s", "3", "test" };
|
|
const config = try parseArgs(allocator, args);
|
|
|
|
try std.testing.expectEqualStrings("test", config.keyword);
|
|
try std.testing.expect(config.target_sections != null);
|
|
try std.testing.expectEqual(@as(usize, 2), config.target_sections.?.len);
|
|
try std.testing.expectEqualStrings("1", config.target_sections.?[0]);
|
|
try std.testing.expectEqualStrings("3", config.target_sections.?[1]);
|
|
}
|
|
|
|
test "parseArgs - error no keyword" {
|
|
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
|
defer _ = gpa.deinit();
|
|
const allocator = gpa.allocator();
|
|
|
|
const args = &[_][:0]u8{"manwhere"};
|
|
const result = parseArgs(allocator, args);
|
|
|
|
try std.testing.expectError(error.NoKeyword, result);
|
|
}
|
|
|
|
test "parseArgs - error invalid option" {
|
|
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
|
defer _ = gpa.deinit();
|
|
const allocator = gpa.allocator();
|
|
|
|
const args = &[_][:0]u8{ "manwhere", "--invalid", "ls" };
|
|
const result = parseArgs(allocator, args);
|
|
|
|
try std.testing.expectError(error.InvalidOption, result);
|
|
}
|
|
|
|
test "parseArgs - error multiple keywords" {
|
|
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
|
defer _ = gpa.deinit();
|
|
const allocator = gpa.allocator();
|
|
|
|
const args = &[_][:0]u8{ "manwhere", "ls", "cat" };
|
|
const result = parseArgs(allocator, args);
|
|
|
|
try std.testing.expectError(error.MultipleKeywords, result);
|
|
}
|