const { keys } = require("./conf.priv.js") const { escape, createSuggestionItem, createURLItem } = require("./util") const completions = {} // Helper functions for Google Custom Search Engine completions const googleCxURL = (alias) => { const key = `google_cx_${alias}` return `https://www.googleapis.com/customsearch/v1?key=${keys.google_cs}&cx=${keys[key]}&q=` } const googleCxPublicURL = (alias) => { const key = `google_cx_${alias}` return `https://cse.google.com/cse/publicurl?cx=${keys[key]}&q=` } const googleCxCallback = (response) => { const res = JSON.parse(response.text).items return res.map(s => createSuggestionItem(`
${s.htmlTitle}
${s.htmlSnippet}
`, { url: s.link })) } // ****** Arch Linux ****** // // Arch Linux official repos completions.al = { alias: "al", name: "archlinux", search: "https://www.archlinux.org/packages/?arch=x86_64&q=", compl: googleCxURL("al"), callback: googleCxCallback, } // Arch Linux AUR completions.au = { alias: "au", name: "AUR", search: "https://aur.archlinux.org/packages/?O=0&SeB=nd&outdated=&SB=v&SO=d&PP=100&do_Search=Go&K=", compl: "https://aur.archlinux.org/rpc?type=suggest&arg=", } completions.au.callback = (response) => { const res = JSON.parse(response.text) return res.map(s => createURLItem(s, `https://aur.archlinux.org/packages/${s}`)) } // Arch Linux Wiki completions.aw = { alias: "aw", name: "archwiki", search: "https://wiki.archlinux.org/index.php?go=go&search=", compl: "https://wiki.archlinux.org/api.php?action=opensearch&format=json&formatversion=2&namespace=0&limit=10&suggest=true&search=", } completions.aw.callback = response => JSON.parse(response.text)[1] // Arch Linux Forums completions.af = { alias: "af", name: "archforums", search: googleCxPublicURL("af"), compl: googleCxURL("af"), callback: googleCxCallback, } // ****** Technical Resources ****** // // Chrome Webstore completions.cs = { alias: "cs", name: "chromestore", search: "https://chrome.google.com/webstore/search/", compl: googleCxURL("cs"), callback: googleCxCallback, } // OWASP Wiki completions.ow = { alias: "ow", name: "owasp", search: "https://www.owasp.org/index.php?go=go&search=", compl: "https://www.owasp.org/api.php?action=opensearch&format=json&formatversion=2&namespace=0&limit=10&suggest=true&search=", } completions.ow.callback = response => JSON.parse(response.text)[1] // StackOverflow completions.so = { alias: "so", name: "stackoverflow", search: "https://stackoverflow.com/search?q=", compl: "https://api.stackexchange.com/2.2/search/advanced?pagesize=10&order=desc&sort=relevance&site=stackoverflow&q=", } completions.so.callback = response => JSON.parse(response.text).items.map(s => createURLItem(`[${s.score}] ${s.title}`, s.link)) // DockerHub repo search completions.dh = { alias: "dh", name: "dockerhub", search: "https://hub.docker.com/search/?page=1&q=", compl: "https://hub.docker.com/v2/search/repositories/?page_size=20&query=", } completions.dh.callback = response => JSON.parse(response.text).results.map((s) => { let meta = "" let repo = s.repo_name meta += `[★${escape(s.star_count)}] ` meta += `[↓${escape(s.pull_count)}] ` if (repo.indexOf("/") === -1) { repo = `_/${repo}` } return createSuggestionItem(`
${escape(repo)}
${meta}
${escape(s.short_description)}
`, { url: `https://hub.docker.com/r/${repo}` }) }) // GitHub completions.gh = { alias: "gh", name: "github", search: "https://github.com/search?q=", compl: "https://api.github.com/search/repositories?sort=stars&order=desc&q=", } completions.gh.callback = response => JSON.parse(response.text).items.map((s) => { let prefix = "" if (s.stargazers_count) { prefix += `[★${s.stargazers_count}] ` } return createURLItem(prefix + s.full_name, s.html_url) }) // Domainr domain search completions.do = { alias: "do", name: "domainr", search: "https://domainr.com/?q=", compl: `https://domainr.p.mashape.com/v2/search?mashape-key=${keys.domainr}&query=%s`, } completions.do.callback = response => JSON.parse(response.text).results .map(d => createSuggestionItem( `
${escape(d.domain)}
`, { url: `https://domainr.com/${d.domain}` } )) // Vim Wiki completions.vw = { alias: "vw", name: "vimwikia", search: "https://vim.wikia.com/wiki/Special:Search?query=", compl: "https://vim.wikia.com/api.php?action=opensearch&format=json&formatversion=2&namespace=0&limit=10&suggest=true&search=", } completions.vw.callback = response => JSON.parse(response.text)[1] .map(r => createURLItem(r, `https://vim.wikia.com/wiki/${r}`)) // ****** Shopping & Food ****** // // Amazon completions.az = { alias: "az", name: "amazon", search: "https://smile.amazon.com/s/?field-keywords=", compl: "https://completion.amazon.com/search/complete?method=completion&mkt=1&search-alias=aps&q=", } completions.az.callback = response => JSON.parse(response.text)[1] // Craigslist completions.cl = { alias: "cl", name: "craigslist", search: "https://craigslist.org/search/sss?query=", compl: "https://craigslist.org/suggest?v=12&type=search&cat=sss&area=1&term=", } completions.cl.callback = response => JSON.parse(response.text) // EBay completions.eb = { alias: "eb", name: "ebay", search: "https://www.ebay.com/sch/i.html?_nkw=", compl: "https://autosug.ebay.com/autosug?callback=0&sId=0&kwd=", } completions.eb.callback = response => JSON.parse(response.text).res.sug // Yelp completions.yp = { alias: "yp", name: "yelp", search: "https://www.yelp.com/search?find_desc=", compl: "https://www.yelp.com/search_suggest/v2/prefetch?prefix=", } completions.yp.callback = (response) => { const res = JSON.parse(response.text).response const words = [] res.forEach((r) => { r.suggestions.forEach((s) => { const w = s.query if (words.indexOf(w) === -1) { words.push(w) } }) }) return words } // ****** General References, Calculators & Utilities ****** // const parseDatamuseRes = (res, o = {}) => { const opts = Object.assign({}, { maxDefs: -1, ellipsis: false, }, o) return res.map((r) => { const defs = [] let defsHtml = "" if ((opts.maxDefs <= -1 || opts.maxDefs > 0) && r.defs && r.defs.length > 0) { for (const d of r.defs.slice(0, opts.maxDefs <= -1 ? undefined : opts.maxDefs)) { const ds = d.split("\t") const partOfSpeech = `(${escape(ds[0])})` const def = escape(ds[1]) defs.push(`${partOfSpeech} ${def}`) } if (opts.ellipsis && r.defs.length > opts.maxDefs) { defs.push("") } defsHtml = `
${defs.join("
")}
` } return createSuggestionItem(`
${escape(r.word)}
${defsHtml}
`, { url: `${opts.wordBaseURL}${r.word}` }) }) } // Dictionary completions.de = { alias: "de", name: "define", search: "http://onelook.com/?w=", compl: "https://api.datamuse.com/words?md=d&sp=%s*", opts: { maxDefs: 16, ellipsis: true, wordBaseURL: "http://onelook.com/?w=", }, } completions.de.callback = (response) => { const res = JSON.parse(response.text) return parseDatamuseRes(res, completions.de.opts) } // Thesaurus completions.th = { alias: "th", name: "thesaurus", search: "https://www.onelook.com/thesaurus/?s=", compl: "https://api.datamuse.com/words?md=d&ml=%s", opts: { maxDefs: 3, ellipsis: true, wordBaseURL: "http://onelook.com/thesaurus/?s=", }, } completions.th.callback = (response) => { const res = JSON.parse(response.text) return parseDatamuseRes(res, completions.th.opts) } // Wikipedia completions.wp = { alias: "wp", name: "wikipedia", search: "https://en.wikipedia.org/w/index.php?search=", compl: "https://en.wikipedia.org/w/api.php?action=query&format=json&generator=prefixsearch&prop=info|pageprops%7Cpageimages%7Cdescription&redirects=&ppprop=displaytitle&piprop=thumbnail&pithumbsize=100&pilimit=6&inprop=url&gpssearch=", } const wpNoimg = "" const parseCrunchbase = (response, parse) => { const res = JSON.parse(response.text).data.items if (res.length === 0) { return [createSuggestionItem(`
No Results
Nothing matched your query
`, { url: "https://www.crunchbase.com/" })] } const objs = res.map(obj => parse(obj)) return objs.map((p) => { const domain = p.domain ? ` | ${p.domain}` : "" const location = p.loc ? ` located in ${p.loc}` : "" return createSuggestionItem(`
${p.name}
${p.name}
${p.desc}
${p.role}${location}${domain}
`, { url: p.url }) }) } // Crunchbase Organization Search completions.co = { alias: "co", name: "crunchbase-orgs", search: "https://www.crunchbase.com/textsearch?q=", compl: `https://api.crunchbase.com/v/3/odm_organizations?user_key=${keys.crunchbase}&query=%s`, } completions.co.callback = response => parseCrunchbase(response, (org) => { const r = org.properties const p = { name: escape(r.name), domain: r.domain !== null ? escape(r.domain).replace(/\/$/, "") : null, desc: escape(r.short_description), role: escape(r.primary_role), img: blank, loc: "", url: `https://www.crunchbase.com/${r.web_path}`, } p.loc += (r.city_name !== null) ? escape(r.city_name) : "" p.loc += (r.region_name !== null && p.loc !== "") ? ", " : "" p.loc += (r.region_name !== null) ? escape(r.region_name) : "" p.loc += (r.country_code !== null && p.loc !== "") ? ", " : "" p.loc += (r.country_code !== null) ? escape(r.country_code) : "" if (r.profile_image_url !== null) { const u = r.profile_image_url const img = u.slice(u.indexOf("t_api_images") + "t_api_images".length + 1) p.img = `https://res-4.cloudinary.com/crunchbase-production/image/upload/c_lpad,h_100,w_100,f_auto,b_white,q_auto:eco/${img}` } return p }) // Crunchbase People Search completions.cp = { alias: "cp", name: "crunchbase-people", search: "https://www.crunchbase.com/app/search/?q=", compl: `https://api.crunchbase.com/v/3/odm_people?user_key=${keys.crunchbase}&query=%s`, } completions.cp.callback = response => parseCrunchbase(response, (person) => { const r = person.properties const p = { name: `${escape(r.first_name)} ${escape(r.last_name)}`, desc: "", img: blank, role: "", loc: "", url: `https://www.crunchbase.com/${r.web_path}`, } p.desc += (r.title !== null) ? escape(r.title) : "" p.desc += (r.organization_name !== null && p.desc !== "") ? ", " : "" p.desc += (r.organization_name !== null) ? escape(r.organization_name) : "" p.loc += (r.city_name !== null) ? escape(r.city_name) : "" p.loc += (r.region_name !== null && p.loc !== "") ? ", " : "" p.loc += (r.region_name !== null) ? escape(r.region_name) : "" p.loc += (r.country_code !== null && p.loc !== "") ? ", " : "" p.loc += (r.country_code !== null) ? escape(r.country_code) : "" if (r.profile_image_url !== null) { const url = r.profile_image_url const path = url.split("/") const img = encodeURIComponent(path[path.length - 1]) p.img = `http://public.crunchbase.com/t_api_images/v1402944794/c_pad,h_50,w_50/${img}` } return p }) // ****** Search Engines ****** // // DuckDuckGo completions.dg = { alias: "dg", name: "duckduckgo", search: "https://duckduckgo.com/?q=", compl: "https://duckduckgo.com/ac/?q=", } completions.dg.callback = response => JSON.parse(response.text).map(r => r.phrase) // Google completions.go = { alias: "go", name: "google", search: "https://www.google.com/search?q=", compl: "https://www.google.com/complete/search?client=chrome-omni&gs_ri=chrome-ext&oit=1&cp=1&pgcl=7&q=", } completions.go.callback = response => JSON.parse(response.text)[1] // Google Images completions.gi = { alias: "gi", name: "google-images", search: "https://www.google.com/search?tbm=isch&q=", compl: "https://www.google.com/complete/search?client=chrome-omni&gs_ri=chrome-ext&oit=1&cp=1&pgcl=7&ds=i&q=", } completions.gi.callback = response => JSON.parse(response.text)[1] // Google - I'm Feeling Lucky completions.gl = { alias: "gl", name: "google-lucky", search: "https://www.google.com/search?btnI=1&q=", compl: "https://www.google.com/complete/search?client=chrome-omni&gs_ri=chrome-ext&oit=1&cp=1&pgcl=7&q=", } completions.gl.callback = response => JSON.parse(response.text)[1] // ****** Elixir ****** // // Hex.pm completions.hx = { alias: "hx", name: "hex", search: "https://hex.pm/packages?sort=downloads&search=", compl: "https://hex.pm/api/packages?sort=downloads&hx&search=", } completions.hx.callback = response => JSON.parse(response.text).map((s) => { let dls = "" let desc = "" let liscs = "" if (s.downloads && s.downloads.all) { dls = `[↓${escape(s.downloads.all)}] ` } if (s.meta) { if (s.meta.description) { desc = escape(s.meta.description) } if (s.meta.licenses) { s.meta.licenses.forEach((l) => { liscs += `[©${escape(l)}] ` }) } } return createSuggestionItem(`
${escape(s.repository)}/${escape(s.name)}
${dls}${liscs}
${desc}
`, { url: s.html_url }) }) // hexdocs // Same as hex but links to documentation pages completions.hd = { alias: "hd", name: "hexdocs", search: "https://hex.pm/packages?sort=downloads&search=", compl: "https://hex.pm/api/packages?sort=downloads&hd&search=", } completions.hd.callback = response => JSON.parse(response.text).map((s) => { let dls = "" let desc = "" if (s.downloads && s.downloads.all) { dls = `[↓${escape(s.downloads.all)}]` } if (s.meta) { if (s.meta.description) { desc = escape(s.meta.description) } } return createSuggestionItem(`
${escape(s.repository)}/${escape(s.name)}${dls}
${desc}
`, { url: `https://hexdocs.pm/${encodeURIComponent(s.name)}` }) }) // Exdocs // Similar to `hd` but searches inside docs using Google Custom Search completions.ex = { alias: "ex", name: "exdocs", search: "https://hex.pm/packages?sort=downloads&ex&search=", compl: googleCxURL("ex"), } completions.ex.callback = response => JSON.parse(response.text).items.map((s) => { let hash = "" const snippet = s.htmlSnippet const openTag = "" const closeTag = "" const openArgs = "(" const closeArgs = ")" let f1 = snippet.indexOf(openTag) if (f1 === -1) { return null } const f2 = snippet.indexOf(closeTag) if (f2 === -1) { return null } f1 += openTag.length const f3 = f2 + closeTag.length const fname = snippet.slice(f1, f2) const snippetEnd = snippet.slice(f3) const a1 = snippetEnd.indexOf(openArgs) if (a1 !== 0) { return null } let a2 = snippetEnd.indexOf(closeArgs) if (a2 === -1) { return null } a2 += closeArgs.length const fargs = snippetEnd.slice(a1, a2) const fary = fargs.replace(new RegExp(openArgs + closeArgs), "").split(",").length hash = escape(`${fname}/${fary}`) const moduleName = escape(s.title).split(" –")[0] let subtitle = "" if (hash) { subtitle = `
${moduleName}.${hash}
` } return createSuggestionItem(`
${s.htmlTitle}
${subtitle}
${s.htmlSnippet}
`, { url: `${s.link}#${hash}` }) }).filter(s => s !== null) // ****** Golang ****** // // Golang Docs (Google CSE) completions.gg = { alias: "gg", name: "golang", search: googleCxPublicURL("gg"), compl: googleCxURL("gg"), callback: googleCxCallback, } // Godoc completions.gd = { alias: "gd", name: "godoc", search: "https://godoc.org/?q=", compl: "https://api.godoc.org/search?q=", } completions.gd.callback = response => JSON.parse(response.text).results.map((s) => { let prefix = "" if (s.import_count) { prefix += `[↓${s.import_count}] ` } if (s.stars) { prefix += `[★${s.stars}] ` } return createURLItem(prefix + s.path, `https://godoc.org/${s.path}`) }) // Gowalker completions.gw = { alias: "gw", name: "gowalker", search: "https://gowalker.org/search?auto_redirect=true&q=", compl: "https://gowalker.org/search/json?q=", } completions.gw.callback = response => JSON.parse(response.text).results.map((s) => { const title = escape(s.title) const desc = escape(s.description) return createSuggestionItem(`
${title}
${desc}
`, { url: `https://golang.org/doc/${encodeURIComponent(s.url)}` }) }) // Go-Search completions.gs = { alias: "gs", name: "go-search", search: "http://go-search.org/search?q=", compl: "http://go-search.org/api?action=search&q=", } completions.gs.callback = response => JSON.parse(response.text).hits .map(r => r.package) // ****** Haskell ****** // // Hackage completions.ha = { alias: "ha", name: "hackage", search: "https://hackage.haskell.org/packages/search?terms=", compl: "https://hackage.haskell.org/packages/search.json?terms=", } completions.ha.callback = response => JSON.parse(response.text) .map(s => createURLItem(s.name, `https://hackage.haskell.org/package/${s.name}`)) // Hoogle completions.ho = { alias: "ho", name: "hoogle", search: `https://www.haskell.org/hoogle/?hoogle=${ encodeURIComponent("+platform +xmonad +xmonad-contrib ")}`, // This tells Hoogle to include these modules in the search - encodeURIComponent is only used for better readability compl: `https://www.haskell.org/hoogle/?mode=json&hoogle=${ encodeURIComponent("+platform +xmonad +xmonad-contrib ")}`, } completions.ho.callback = response => JSON.parse(response.text).results .map(s => createURLItem(s.self, s.location)) // Haskell Wiki completions.hw = { alias: "hw", name: "haskellwiki", search: "https://wiki.haskell.org/index.php?go=go&search=", compl: "https://wiki.haskell.org/api.php?action=opensearch&format=json&formatversion=2&namespace=0&limit=10&suggest=true&search=", } completions.hw.callback = response => JSON.parse(response.text)[1] // Hayoo completions.hy = { alias: "hy", name: "hayoo", search: "http://hayoo.fh-wedel.de/?query=", compl: "http://hayoo.fh-wedel.de/json?query=", } completions.hy.callback = response => JSON.parse(response.text).result .map(s => createURLItem(`[${s.resultType}] ${s.resultName}`, s.resultUri)) // ****** HTML, CSS, JavaScript, NodeJS, ... ****** // // jQuery API documentation completions.jq = { alias: "jq", name: "jquery", search: googleCxPublicURL("jq"), compl: googleCxURL("jq"), callback: googleCxCallback, } // NodeJS standard library documentation completions.no = { alias: "no", name: "node", search: googleCxPublicURL("no"), compl: googleCxURL("no"), callback: googleCxCallback, } // Mozilla Developer Network (MDN) completions.md = { alias: "md", name: "mdn", search: "https://developer.mozilla.org/en-US/search?q=", compl: "https://developer.mozilla.org/en-US/search.json?q=", } completions.md.callback = (response) => { const res = JSON.parse(response.text) return res.documents.map((s) => { let excerpt = escape(s.excerpt) if (excerpt.length > 240) { excerpt = `${excerpt.slice(0, 240)}…` } res.query.split(" ").forEach((q) => { excerpt = excerpt.replace(new RegExp(q, "gi"), "$&") }) const title = escape(s.title) const slug = escape(s.slug) return createSuggestionItem(`
${title}
${slug}
${excerpt}
`, { url: s.url }) }) } // NPM registry search completions.np = { alias: "np", name: "npm", search: "https://www.npmjs.com/search?q=", compl: "https://api.npms.io/v2/search/suggestions?size=20&q=", } completions.np.callback = response => JSON.parse(response.text) .map((s) => { let flags = "" let desc = "" let stars = "" let score = "" if (s.package.description) { desc = escape(s.package.description) } if (s.score) { if (s.score.final) { score = Math.round(Number(s.score.final) * 5) stars = "★".repeat(score) + "☆".repeat(5 - score) } } if (s.flags) { Object.keys(s.flags).forEach((f) => { flags += `[ ${escape(f)}] ` }) } return createSuggestionItem(`
${s.highlight}
${stars} ${flags}
${desc}
`, { url: s.package.links.npm }) }) // ****** Social Media & Entertainment ****** // // Hacker News (YCombinator) completions.hn = { alias: "hn", name: "hackernews", search: "https://hn.algolia.com/?query=", compl: "https://hn.algolia.com/api/v1/search?tags=(story,comment)&query=", } completions.hn.callback = (response) => { const res = JSON.parse(response.text) return res.hits.map((s) => { let title = "" let prefix = "" if (s.points) { prefix += `[↑${s.points}] ` } if (s.num_comments) { prefix += `[↲${s.num_comments}] ` } switch (s._tags[0]) { // eslint-disable-line no-underscore-dangle case "story": title = s.title // eslint-disable-line prefer-destructuring break case "comment": title = s.comment_text break default: title = s.objectID } const re = new RegExp(`(${res.query.split(" ").join("|")})`, "ig") title = title.replace(re, "$&") const url = `https://news.ycombinator.com/item?id=${s.objectID}` return createSuggestionItem(`
${prefix + title}
${url}
`, { url }) }) } // Reddit completions.re = { alias: "re", name: "reddit", search: "https://www.reddit.com/search?sort=relevance&t=all&q=", compl: "https://api.reddit.com/search?syntax=plain&sort=relevance&limit=20&q=", } completions.re.callback = response => JSON.parse(response.text).data.children .map(s => createURLItem(`[${s.data.score}] ${s.data.title}`, `https://reddit.com${s.data.permalink}`)) // YouTube completions.yt = { alias: "yt", name: "youtube", search: "https://www.youtube.com/search?q=", compl: `https://www.googleapis.com/youtube/v3/search?maxResults=20&part=snippet&type=video,channel&key=${keys.google_yt}&safeSearch=none&q=`, } completions.yt.callback = response => JSON.parse(response.text).items .map((s) => { switch (s.id.kind) { case "youtube#channel": return createURLItem( `${s.snippet.channelTitle}: ${s.snippet.description}`, `https://youtube.com/channel/${s.id.channelId}`, ) case "youtube#video": return createURLItem( ` ▶ ${s.snippet.title}`, `https://youtu.be/${s.id.videoId}`, ) default: return null } }).filter(s => s !== null) module.exports = completions