Skip to content

Commit 5d0ec63

Browse files
James Raubenheimerjamzrob
James Raubenheimer
authored andcommitted
feat(companies): Leet companies - view company list to filter problems by
1 parent a92e764 commit 5d0ec63

File tree

13 files changed

+378
-1
lines changed

13 files changed

+378
-1
lines changed

README.md

+2
Original file line numberDiff line numberDiff line change
@@ -360,6 +360,8 @@ image_support = false,
360360
- `daily` opens the question of today
361361

362362
- `list` opens a problem list picker
363+
-
364+
- `companies` opens a list of companies to filter questions by
363365

364366
- `open` opens the current question in a default browser
365367

lua/leetcode-plugins/cn/queries.lua

+10
Original file line numberDiff line numberDiff line change
@@ -160,3 +160,13 @@ queries.session_progress = [[
160160
}
161161
}
162162
]]
163+
164+
queries.streak = [[
165+
query questionCompanyTags {
166+
companyTags {
167+
name
168+
slug
169+
questionCount
170+
}
171+
}
172+
]]

lua/leetcode-ui/group/page/problems.lua

+7
Original file line numberDiff line numberDiff line change
@@ -33,12 +33,19 @@ local daily = Button("Daily", {
3333
on_press = cmd.qot,
3434
})
3535

36+
local companies = Button("By Company", {
37+
icon = "",
38+
sc = "c",
39+
on_press = cmd.companies,
40+
})
41+
3642
local back = BackButton("menu")
3743

3844
page:insert(Buttons({
3945
list,
4046
random,
4147
daily,
48+
companies,
4249
back,
4350
}))
4451

lua/leetcode/api/companies.lua

+81
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
local utils = require("leetcode.api.utils")
2+
local config = require("leetcode.config")
3+
local log = require("leetcode.logger")
4+
local queries = require("leetcode.api.queries")
5+
local urls = require("leetcode.api.urls")
6+
local Spinner = require("leetcode.logger.spinner")
7+
8+
---@class lc.CompaniesApi
9+
local Companies = {}
10+
11+
---@param cb? fun(res: lc.cache.Company[]|nil, err: lc.err)
12+
---@param noti? boolean
13+
--
14+
---@return lc.cache.Company[] lc.err
15+
function Companies.all(cb, noti)
16+
local query = queries.companies
17+
18+
local spinner
19+
if noti then
20+
spinner = Spinner:init("updating cache...", "points")
21+
end
22+
if cb then
23+
utils.query(query, _, {
24+
endpoint = urls.companies,
25+
callback = function(res, err)
26+
if err then
27+
if spinner then
28+
spinner:stop(err.msg, false)
29+
end
30+
return cb(nil, err)
31+
end
32+
local data = res.data
33+
local companies = data["companyTags"]
34+
if spinner then
35+
spinner:stop("cache updated")
36+
end
37+
cb(companies)
38+
end,
39+
})
40+
else
41+
local res, err = utils.query(query)
42+
if err then
43+
if spinner then
44+
spinner:stop(err.msg, false)
45+
end
46+
return nil, err
47+
else
48+
local data = res.data
49+
local companies = data["companyTags"]
50+
if spinner then
51+
spinner:stop("cache updated")
52+
end
53+
return companies
54+
end
55+
end
56+
end
57+
58+
function Companies.problems(company, cb)
59+
local url = urls.company_problems:format(company)
60+
61+
if cb then
62+
utils.get(url, {
63+
callback = function(res, err)
64+
if err then
65+
return cb(nil, err)
66+
end
67+
local questions = res["questions"]
68+
cb(questions)
69+
end
70+
})
71+
else
72+
local res, err = utils.get(url)
73+
if err then
74+
return nil, err
75+
end
76+
local questions = res.data["questions"]
77+
return questions
78+
end
79+
end
80+
81+
return Companies

lua/leetcode/api/problems.lua

+1
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ function Problems.all(cb, noti)
3131
return cb(nil, err)
3232
end
3333

34+
3435
local problems = utils.normalize_problems(res.stat_status_pairs)
3536

3637
if config.is_cn then

lua/leetcode/api/queries.lua

+10
Original file line numberDiff line numberDiff line change
@@ -179,4 +179,14 @@ queries.session_progress = [[
179179
}
180180
]]
181181

182+
queries.companies = [[
183+
query questionCompanyTags {
184+
companyTags {
185+
name
186+
slug
187+
questionCount
188+
}
189+
}
190+
]]
191+
182192
return queries

lua/leetcode/api/types.lua

+6
Original file line numberDiff line numberDiff line change
@@ -359,6 +359,12 @@
359359
---@field questions_count table<lc.Stats.QuestionCount>
360360
---@field submit_stats lc.Stats.SubmissionStat
361361

362+
--------------------------------------------
363+
--- Companies
364+
--------------------------------------------
365+
---@class lc.Companies.Res
366+
---@field companyTags {name: string, slug: string, questionCount: number}[]
367+
362368
--------------------------------------------
363369
--- Skills
364370
--------------------------------------------

lua/leetcode/api/urls.lua

+2
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,12 @@ urls.base = "/graphql/"
55
urls.solved = "/graphql/"
66
urls.calendar = "/graphql/"
77
urls.languages = "/graphql/"
8+
urls.companies = "/graphql/"
89
urls.skills = "/graphql/"
910
urls.auth = "/graphql/"
1011

1112
urls.problems = "/api/problems/%s/"
13+
urls.company_problems = "/problems/tag-data/company-tags/%s/"
1214
urls.interpret = "/problems/%s/interpret_solution/"
1315
urls.submit = "/problems/%s/submit/"
1416
urls.run = "/problems/%s/interpret_solution/"

lua/leetcode/api/utils.lua

+23
Original file line numberDiff line numberDiff line change
@@ -248,4 +248,27 @@ function utils.translate_titles(problems, titles)
248248
end, problems)
249249
end
250250

251+
--@return lc.cache.Question[]
252+
function utils.normalize_company_problems(problems)
253+
return vim.tbl_map(function(p)
254+
return {
255+
status = p.status,
256+
id = tonumber(p.questionId),
257+
frontend_id = tonumber(p.questionFrontendId),
258+
title = p.title,
259+
title_cn = "",
260+
title_slug = p.titleSlug,
261+
link = ("https://leetcode.%s/problems/%s/"):format(
262+
config.domain,
263+
p.titleSlug
264+
),
265+
paid_only = p.isPaidOnly,
266+
ac_rate = tonumber(p.acRate:sub(1, -2)),
267+
difficulty = p.difficulty,
268+
starred = false,
269+
topic_tags = {},
270+
}
271+
end, problems)
272+
end
273+
251274
return utils

lua/leetcode/cache/companylist.lua

+123
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,123 @@
1+
local path = require("plenary.path")
2+
local companies_api = require("leetcode.api.companies")
3+
4+
local log = require("leetcode.logger")
5+
local config = require("leetcode.config")
6+
local interval = config.user.cache.update_interval
7+
8+
---@type Path
9+
local file = config.storage.cache:joinpath(("companylist%s"):format(config.is_cn and "_cn" or ""))
10+
11+
---@type { at: integer, payload: lc.cache.payload }
12+
local hist = nil
13+
14+
---@class lc.cache.Company
15+
---@field name string
16+
---@field slug string
17+
---@field questionCount number
18+
19+
20+
---@class lc.cache.Copmanylist
21+
local Companylist = {}
22+
23+
---@return lc.cache.Company[]
24+
function Companylist.get()
25+
return Companylist.read().data
26+
end
27+
28+
---@return lc.cache.payload
29+
function Companylist.read()
30+
if not file:exists() then
31+
return Companylist.populate()
32+
end
33+
34+
local time = os.time()
35+
if hist and (time - hist.at) <= math.min(60, interval) then
36+
return hist.payload
37+
end
38+
39+
local contents = file:read()
40+
if not contents or type(contents) ~= "string" then
41+
return Companylist.populate()
42+
end
43+
44+
local cached = Companylist.parse(contents)
45+
46+
if not cached or (cached.version ~= config.version or cached.username ~= config.auth.name) then
47+
return Companylist.populate()
48+
end
49+
50+
hist = { at = time, payload = cached }
51+
if (time - cached.updated_at) > interval then
52+
Companylist.update()
53+
end
54+
55+
return cached
56+
end
57+
58+
---@return lc.cache.payload
59+
function Companylist.populate()
60+
local res, err = companies_api.all(nil, true)
61+
62+
if not res or err then
63+
local msg = (err or {}).msg or "failed to fetch company list"
64+
error(msg)
65+
end
66+
67+
Companylist.write({ data = res })
68+
return hist.payload
69+
end
70+
71+
function Companylist.update()
72+
companies_api.all(function(res, err)
73+
if not err then
74+
Companylist.write({ data = res })
75+
end
76+
end, true)
77+
end
78+
79+
---@return lc.cache.Company
80+
function Companylist.get_by_title_slug(title_slug)
81+
local companies = Companylist.get()
82+
83+
local company = vim.tbl_filter(function(e)
84+
return e.title_slug == slug
85+
end, companies)[1]
86+
87+
assert(company("Company `%s` not found. Try updating cache?"):format(title_slug))
88+
return company
89+
end
90+
91+
---@param payload? lc.cache.payload
92+
function Companylist.write(payload)
93+
payload = vim.tbl_deep_extend("force", {
94+
version = config.version,
95+
updated_at = os.time(),
96+
username = config.auth.name,
97+
}, payload)
98+
99+
if not payload.data then
100+
payload.data = Companylist.get()
101+
end
102+
103+
file:write(vim.json.encode(payload), "w")
104+
hist = { at = os.time(), payload = payload }
105+
end
106+
107+
---@alias lc.cache.payload { version: string, data: lc.cache.Company[], updated_at: integer, username: string }
108+
109+
---@param str string
110+
---
111+
---@return lc.cache.payload
112+
function Companylist.parse(str)
113+
return vim.json.decode(str)
114+
end
115+
116+
function Companylist.delete()
117+
if not file:exists() then
118+
return false
119+
end
120+
return pcall(path.rm, file)
121+
end
122+
123+
return Companylist

lua/leetcode/cache/init.lua

+2
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
11
local Problemlist = require("leetcode.cache.problemlist")
2+
local Companylist = require("leetcode.cache.companylist")
23

34
---@class lc.Cache
45
local cache = {}
56

67
function cache.update()
78
Problemlist.update()
9+
Companylist.update()
810
end
911

1012
return cache

lua/leetcode/command/init.lua

+12-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,14 @@ function cmd.problems(options)
2626
require("leetcode.utils").auth_guard()
2727

2828
local p = require("leetcode.cache.problemlist").get()
29-
require("leetcode.pickers.question").pick(p, options)
29+
end
30+
31+
---@param options table<string, string[]>
32+
function cmd.companies(options)
33+
require("leetcode.utils").auth_guard()
34+
35+
local c = require("leetcode.cache.companylist").get()
36+
require("leetcode.pickers.company").pick(c, options)
3037
end
3138

3239
---@param cb? function
@@ -635,6 +642,10 @@ cmd.commands = {
635642
cmd.problems,
636643
_args = arguments.list,
637644
},
645+
companies = {
646+
cmd.companies,
647+
_args = arguments.list,
648+
},
638649
random = {
639650
cmd.random_question,
640651
_args = arguments.random,

0 commit comments

Comments
 (0)