Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat: Filtering Problem List using local files. #145

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
127 changes: 127 additions & 0 deletions lua/leetcode/cache/problemlist.lua
Original file line number Diff line number Diff line change
Expand Up @@ -142,4 +142,131 @@ function Problemlist.delete()
return pcall(path.rm, file)
end

---@class lc.filter.CustomList
---@field problems { id: string, tags: string[] }[] List of problem definitions
---@field source string Source file path
local function parse_custom_list(filepath)
local expanded_path = vim.fn.expand(filepath)

-- Check file existence
if not vim.fn.filereadable(expanded_path) then
error(("File does not exist or is not readable: %s"):format(expanded_path))
end

local file = io.open(expanded_path, "r")
if not file then
error(("Cannot open custom filter file: %s"):format(filepath))
end

local problems = {}
local file_ext = filepath:match("%.([^%.]+)$")

if file_ext == "json" then
local content = file:read("*all")
file:close()

local ok, decoded = pcall(vim.json.decode, content)
if not ok then
error(("Invalid JSON in file: %s"):format(filepath))
end

-- Support both simple arrays and structured format
if type(decoded) == "table" then
if decoded[1] then -- Simple array format
for _, item in ipairs(decoded) do
problems[#problems + 1] = {
id = tostring(item),
tags = {}
}
end
else -- Structured format
for group, items in pairs(decoded) do
for _, item in ipairs(items) do
problems[#problems + 1] = {
id = tostring(item),
tags = { group }
}
end
end
end
end
else -- Assume text file
for line in file:lines() do
line = line:match("^%s*(.-)%s*$")
if line ~= "" and not line:match("^#") then
local id, tags = line:match("([^|]+)|?(.*)")
if id then
problems[#problems + 1] = {
id = id:match("^%s*(.-)%s*$"),
tags = vim.split(tags, ",")
}
end
end
end
file:close()
end

return {
problems = problems,
source = filepath
}
end

---@param problems lc.cache.Question[]
---@param custom_list lc.filter.CustomList
---@param options lc.filter.Options
---@return lc.cache.Question[]
local function filter_by_custom_list(problems, custom_list, options)
local filtered = {}
local problem_map = {}

-- Create lookup maps for both title slugs and problem IDs
for _, problem in ipairs(problems) do
problem_map[problem.title_slug] = problem
problem_map[problem.frontend_id] = problem
end

-- Filter problems based on the custom list
for _, item in ipairs(custom_list.problems) do
local problem = problem_map[item.id]
if problem then
-- Apply additional filters if specified
if options then
local matches = true

-- Apply difficulty filter
if options.difficulty and problem.difficulty ~= options.difficulty then
matches = false
end

-- Apply status filter
if matches and options.status and options.status ~= "any" then
if options.status == "ac" and problem.status ~= "ac" then
matches = false
elseif options.status == "notac" and problem.status == "ac" then
matches = false
end
end

if matches then
table.insert(filtered, problem)
end
else
table.insert(filtered, problem)
end
end
end

return filtered
end

---@param filepath string Path to custom filter file
---@param options? lc.filter.Options Additional filters to apply
---@return lc.cache.Question[]
function Problemlist.filter_by_file(filepath, options)
local problems = Problemlist.get()
local custom_list = parse_custom_list(filepath)
return filter_by_custom_list(problems, custom_list, options)
end

return Problemlist
5 changes: 5 additions & 0 deletions lua/leetcode/command/arguments.lua
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,11 @@ arguments.random = {
tags = topics,
}

arguments.list_custom = {
file = {}, -- Will be populated with available .txt/.json files
filter = { "topics", "difficulty", "status" }, -- Allow combining with existing filters
}

arguments.session_change = {
name = config.sessions.names,
}
Expand Down
27 changes: 27 additions & 0 deletions lua/leetcode/command/init.lua
Original file line number Diff line number Diff line change
Expand Up @@ -492,6 +492,29 @@ function cmd.fix()
vim.cmd("qa!")
end

function cmd.custom_list(options)
require("leetcode.utils").auth_guard()

if not options.file or #options.file == 0 then
return log.error("No file specified for custom list")
end

local filepath = options.file[1]
local filter_options = {}

-- Apply additional filters if specified
if options.filter then
for _, filter_type in ipairs(options.filter) do
if options[filter_type] then
filter_options[filter_type] = options[filter_type][1]
end
end
end

local problems = require("leetcode.cache.problemlist").filter_by_file(filepath, filter_options)
require("leetcode.pickers.question").pick(problems)
end

---@return string[], string[]
function cmd.parse(args)
local parts = vim.split(vim.trim(args), "%s+")
Expand Down Expand Up @@ -664,6 +687,10 @@ cmd.commands = {
cmd.fix,
_private = true,
},
custom = {
cmd.custom_list,
_args = arguments.list_custom,
},
}

return cmd