FOSS Git Repository & NPM Package Index

Hint: you can search by multiple keywords, separated by space, e.g. react event as searching for repos containing react and event in the name (appearing any order).

Hint: you can indicate negative keywords with hyphen prefix, e.g. -react -ng- chart as searching for chart libraries while excluding those framework-specific libraries having react or ng- in the name.

Hint: multiple keywords are combined with "and" for most fields, but they're combined with "or" for programming languages.

Hint: the keyboards are matched partially for most fields, but is matched exactly for programming languages. So searching Java will not match Javascript repos.

Hint: if a keyword is wrapped with double quotes, it is matched in full. For example searching speed will matched for frank-dspeed but searching "speed" will not match for that user. This feature does not apply to the language field as it's always matched in full.

247,901 matches

pattern: -* (27 matches)
pattern: 0* (61 matches)
pattern: 1* (67 matches)
pattern: 2* (49 matches)
pattern: 3* (55 matches)
pattern: 4* (32 matches)
pattern: 5* (15 matches)
pattern: 6* (9 matches)
pattern: 7* (9 matches)
pattern: 8* (13 matches)
pattern: 9* (6 matches)
pattern: @* (66783 matches)
pattern: _* (999 matches)
pattern: a* (10174 matches)
pattern: b* (8139 matches)
pattern: c* (12497 matches)
pattern: d* (6690 matches)
pattern: e* (10900 matches)
pattern: f* (6168 matches)
pattern: g* (10030 matches)
pattern: h* (4699 matches)
pattern: i* (4743 matches)
pattern: j* (4803 matches)
pattern: k* (2826 matches)
pattern: l* (5307 matches)
pattern: m* (9835 matches)
pattern: n* (11545 matches)
pattern: o* (3348 matches)
pattern: p* (10362 matches)
pattern: q* (1153 matches)
pattern: r* (12160 matches)
pattern: s* (17207 matches)
pattern: t* (10292 matches)
pattern: u* (3435 matches)
pattern: v* (4568 matches)
pattern: w* (5088 matches)
pattern: x* (1117 matches)
pattern: y* (1430 matches)
pattern: z* (1260 matches)
Source Code of home.tsx
(import statements omitted for simplicity, click to expand)
import { o } from '../jsx/jsx.js'
import SourceCode from '../components/source-code.js'
import { mapArray } from '../components/fragment.js'
import { DynamicContext } from '../context.js'
import Style from '../components/style.js'
import { db } from '../../../db/db.js'
import { Script } from '../components/script.js'
import { EarlyTerminate } from '../helpers.js'
import { ProgrammingLanguageSpan } from '../components/programming-language.js'
import { Link } from '../components/router.js'
import { nodeToVNode } from '../jsx/vnode.js'
import { Element } from '../jsx/types.js'
import { newDB } from 'better-sqlite3-schema'
import { DAY } from '@beenotung/tslib/time.js'
import { Routes } from '../routes.js'
import { prepared_statement_cache, query_cache, sql_cache } from '../cache.js'
// Calling <Component/> will transform the JSX into AST for each rendering.
// You can reuse a pre-compute AST like `let component = <Component/>`.

// If the expression is static (not depending on the render Context),
// you don't have to wrap it by a function at all.

let style = Style(/* css */ `
#searchForm label {
  display: block;
  width: 100%;
  margin: 0.25rem;
}
.hint {
  border-inline-start: 3px solid #748;
  background-color: #edf;
  padding: 1rem;
  margin: 0.5rem 0;
  width: fit-content;
}
.hint code {
  background-color: #fef;
  outline: 1px solid #aaa;
  border-radius: 0.25rem;
  padding: 0.1rem;
  display: inline-block;
}
.hide-hints .hint,
.hide-hints #hideHintsBtn
{
  display: none;
}
#showHintsBtn {
  display: none;
}
.hide-hints #showHintsBtn {
  display: block;
}
.list {
  padding: 0.25rem;
}
.res-group,
.res {
  padding: 0.25rem;
  padding-bottom: 0.5rem;
}
.res-desc {
  margin-top: 0.25rem;
}
`)

let script = Script(/* javascript */ `
function autoFocusKeyword() {
  if (searchForm?.keyword) {
    searchForm.keyword.focus()
    return
  }
  setTimeout(autoFocusKeyword, 33)
}
autoFocusKeyword()

function hideHints() {
  let hide_interval = +localStorage.getItem('hide_interval') || 0
  if (!hide_interval) {
    hide_interval = 1
  } else {
    hide_interval *= 1.5
  }
  localStorage.setItem('hide_interval', hide_interval)
  let hide_hint_until = Date.now() + hide_interval * ${DAY}
  localStorage.setItem('hide_hint_until', hide_hint_until)
  searchForm.classList.add('hide-hints')
}
function showHints() {
  localStorage.removeItem('hide_hint_until')
  localStorage.removeItem('hide_interval')
  searchForm.classList.remove('hide-hints')
}
function autoHideHints() {
  console.log('autoHideHints')
  let hide_hint_until = +localStorage.getItem('hide_hint_until')
  if (Date.now() < hide_hint_until) {
    searchForm.classList.add('hide-hints')
  }
}
autoHideHints()
`)

type MatchedNpmPackage = {
  name: string
  username: string
  desc: string | null
  programming_language: string | null
  weekly_downloads: number | null
  deprecated: number | null
}

type MatchedItem = {
  name: string
  desc: string | null
  url: string
  programming_language: string | null
  username: string
  weekly_downloads: number | null
  is_fork: number | null
  deprecated: number | null
}

function build_search_query(params: URLSearchParams) {
  let action = params.get('form_action')
  let host = params.get('host')
  let username = params.get('username')
  let name = params.get('name')
  let language = params.get('language')
  let desc = params.get('desc')
  let prefix = params.get('prefix')

  let search_repo_bindings: Record<string, string> = {}
  let search_repo_bind_count = 0

  let search_npm_package_bindings: Record<string, string> = {}
  let search_npm_package_bind_count = 0

  let search_repo_sql = /* sql */ `
select
  repo.name
, repo.desc
, repo.url
, ifnull(
    programming_language.name,
    case npm_package.has_types
      when 1 then 'Typescript'
      when 0 then 'Javascript'
    end)
  as programming_language
, author.username
, repo.is_fork
, npm_package.deprecated
from repo
inner join author on author.id = repo.author_id
inner join domain on domain.id = repo.domain_id
left join programming_language on programming_language.id = repo.programming_language_id
left join npm_package on npm_package.repo_id = repo.id
where repo.is_public = 1
`

  let search_npm_package_sql = /* sql */ `
select
  npm_package.name
, author.username
, npm_package.desc
, npm_package.weekly_downloads
, case npm_package.has_types
    when 1 then 'Typescript'
    when 0 then 'Javascript'
  end as programming_language
, npm_package.deprecated
from npm_package
left join author on author.id = author_id
where repo_id is null
`

  if (prefix) {
    search_repo_sql += /* sql */ `
  and repo.name like :prefix
`
    search_npm_package_sql += /* sql */ `
  and npm_package.name like :prefix
`
    search_repo_bindings.prefix = prefix.toLowerCase() + '%'
    search_npm_package_bindings.prefix = prefix.toLowerCase() + '%'
  }

  // add bindings for search_repo
  let qs: [string | null, string][] = [
    [host, 'domain.host'],
    [username, 'author.username'],
    [name, 'repo.name'],
    [desc, 'repo.desc'],
  ]
  for (let [value, field] of qs) {
    if (value) {
      for (let part of value.split(' ')) {
        part = part.trim()
        if (!part) continue
        search_repo_bind_count++
        let bind = 'b' + search_repo_bind_count
        if (part[0] == '-') {
          part = part.slice(1)
          search_repo_sql += /* sql */ `
  and ${field} not like :${bind}
`
        } else {
          search_repo_sql += /* sql */ `
  and ${field} like :${bind}
`
        }
        if (part.startsWith('"') && part.endsWith('"')) {
          search_repo_bindings[bind] = part.slice(1, -1)
        } else {
          search_repo_bindings[bind] = '%' + part + '%'
        }
      }
    }
  }

  // add bindings for search_npm_package
  qs = [
    [username, 'author.username'],
    [name, 'npm_package.name'],
    [desc, 'npm_package.desc'],
  ]
  for (let [value, field] of qs) {
    if (value) {
      for (let part of value.split(' ')) {
        search_npm_package_bind_count++
        let bind = 'b' + search_npm_package_bind_count
        if (part[0] == '-') {
          part = part.slice(1)
          search_npm_package_sql += /* sql */ `
  and ${field} not like :${bind}
`
        } else {
          search_npm_package_sql += /* sql */ `
  and ${field} like :${bind}
`
        }
        if (part.startsWith('"') && part.endsWith('"')) {
          search_npm_package_bindings[bind] = part.slice(1, -1)
        } else {
          search_npm_package_bindings[bind] = '%' + part + '%'
        }
      }
    }
  }
  let skip_npm = false
  if (host) {
    let include_npm = false
    let include_other = false
    for (let part of host.split(' ')) {
      part = part.trim()
      if (!part) continue
      if (part.startsWith('-npm')) {
        skip_npm = true
        break
      }
      if (part.startsWith('npm')) {
        include_npm = true
        continue
      }
      include_other = true
    }
    skip_npm ||= include_other && !include_npm
  }

  // set search bindings for programming languages
  if (language) {
    let positive_languages: string[] = []
    let negative_languages: string[] = []
    for (let name of language.split(' ')) {
      if (!name) continue
      if (name[0] == '-') {
        name = name.slice(1)
        negative_languages.push(name)
      } else {
        positive_languages.push(name)
      }
    }
    if (positive_languages.length > 0) {
      search_repo_sql += /* sql */ `
  and (${positive_languages
    .map(name => {
      search_repo_bind_count++
      let bind = 'b' + search_repo_bind_count
      search_repo_bindings[bind] = name
      return `programming_language like :${bind}`
    })
    .join(' or ')})
`
    }
    if (negative_languages.length > 0) {
      for (let name of negative_languages) {
        search_repo_bind_count++
        let bind = 'b' + search_repo_bind_count
        search_repo_bindings[bind] = name
        search_repo_sql += /* sql */ `
  and (programming_language not like :${bind} or programming_language is null)
`
      }
    }
  }

  // avoid duplicated records due to join tables
  search_repo_sql += /* sql */ `
group by repo.id
`
  search_npm_package_sql += /* sql */ `
group by npm_package.id
`

  return {
    search_repo_sql,
    search_repo_bindings,
    skip_npm,
    search_npm_package_sql,
    search_npm_package_bindings,
    /* from params */
    action,
    host,
    username,
    name,
    language,
    desc,
    prefix,
  }
}

function cached_query<T = unknown>(sql: string, bindings: object): T[] {
  let sql_index = sql_cache.getIndex(sql)
  let key = sql_index + ':' + JSON.stringify(bindings)
  let value = query_cache.get(key)
  if (!value) {
    let start_time = Date.now()
    console.log('== query ==')
    console.log('key:', key)
    // console.log('bindings:', bindings)
    // console.log('---')
    // console.log(sql)
    // console.log('---')
    value = prepared_statement_cache.get<{}, T>(sql).all(bindings)
    let used_time = Date.now() - start_time
    query_cache.set({ key, value, used_time })
    console.log('used time:', used_time)
    // console.log('====')
  }
  return value
}

function Page(attrs: {}, context: SearchContext) {
  let { params, query } = context
  let { prefix } = query

  let matchedItems = cached_query<MatchedItem>(
    query.search_repo_sql,
    query.search_repo_bindings,
  ).slice()

  let matchedPackages = query.skip_npm
    ? []
    : cached_query<MatchedNpmPackage>(
        query.search_npm_package_sql,
        query.search_npm_package_bindings,
      )
  for (let npm_package of matchedPackages) {
    // FIXME move to render part to avoid bug when collapsed into prefix pattern?
    let { name, username } = npm_package
    if (!username && name.startsWith('@')) {
      username = name.split('/')[0].substring(1)
      npm_package.username = username
    }
    matchedItems.push({
      name,
      desc: npm_package.desc,
      url: `https://www.npmjs.com/package/${npm_package.name}`,
      programming_language: npm_package.programming_language,
      username,
      weekly_downloads: npm_package.weekly_downloads,
      is_fork: null,
      deprecated: npm_package.deprecated,
    })
  }

  let match_count = matchedItems.length

  type Match =
    | { type: 'item'; item: MatchedItem; sortKey: string }
    | { type: 'group'; group: Group; sortKey: string }
  let matches: Match[]

  type Group = {
    prefix: string
    resItems: MatchedItem[]
  }

  let total_match_threshold = 36
  let group_match_threshold = 5
  if (match_count > total_match_threshold) {
    let prefix_length = prefix ? prefix.length + 1 : 1
    let groupDict: Record<string, Group> = {}
    for (let repo of matchedItems) {
      let prefix = repo.name.slice(0, prefix_length).toLowerCase()
      let group = groupDict[prefix]
      if (!group) {
        group = { prefix, resItems: [repo] }
        groupDict[prefix] = group
      } else {
        group.resItems.push(repo)
      }
    }
    matches = Object.values(groupDict).map(group => ({
      type: 'group',
      group,
      sortKey: group.prefix.toLowerCase(),
    }))
  } else {
    matches = matchedItems.map(item => ({
      type: 'item',
      item,
      sortKey: item.name.toLowerCase(),
    }))
  }
  matches = matches
    .flatMap((match): Match[] | Match => {
      if (
        match.type != 'group' ||
        match.group.resItems.length > group_match_threshold
      )
        return match
      return match.group.resItems.map(item => ({
        type: 'item',
        item,
        sortKey: item.name,
      }))
    })
    .sort((a, b) => {
      if (a.type == 'group' && b.type != 'group') return +1
      if (a.type != 'group' && b.type == 'group') return -1
      if (a.sortKey < b.sortKey) return -1
      if (a.sortKey > b.sortKey) return +1
      return 0
    })

  let result: Element = [
    'div#result',
    {},
    [
      <p id="loadingMessage"></p>,
      prefix ? <p>repo/package prefix: {prefix}*</p> : null,
      <p>{match_count.toLocaleString()} matches</p>,
      <div class="list">
        {mapArray(matches, match => {
          if (match.type == 'item') {
            return MatchedItem(match.item)
          }
          let {
            group: { prefix, resItems: repos },
          } = match
          let count = repos.length
          if (count == 1) {
            return MatchedItem(repos[0])
          }
          params.set('prefix', prefix)
          let href = '/?' + params
          return (
            <div class="res-group">
              <Link href={href}>pattern: {prefix}*</Link> ({count} matches)
            </div>
          )
        })}
      </div>,
    ],
  ]
  if (query.action == 'search' && context.type == 'ws') {
    context.ws.send(['update', nodeToVNode(result, context)])
    throw EarlyTerminate
  }
  return (
    <form
      id="searchForm"
      onsubmit="emitForm(event); loadingMessage.textContent='searching...'"
    >
      <input name="form_action" value="search" hidden />
      <label>
        Repo Host:{' '}
        <input name="host" placeholder="e.g. npmjs" value={query.host} />
      </label>
      <label>
        Username:{' '}
        <input
          name="username"
          placeholder="e.g. beeno"
          value={query.username}
        />
      </label>
      <label>
        Repo/Package name:{' '}
        <input
          name="name"
          placeholder={'e.g. react event'}
          value={query.name}
        />
      </label>
      <label>
        Programming Languages:{' '}
        <input
          name="language"
          placeholder={'e.g. typescript javascript'}
          value={query.language}
        />
      </label>
      <label>
        Description:{' '}
        <input name="desc" placeholder={'e.g. 倉頡'} value={query.desc} />
      </label>
      <input type="submit" value="Search" />
      <div style="margin-top: 0.5rem">
        <button
          type="button"
          id="hideHintsBtn"
          onclick="hideHints()"
          title="Hide hints for 24 hours"
        >
          Hide Hints
        </button>
        <button type="button" id="showHintsBtn" onclick="showHints()">
          Show Hints
        </button>
      </div>
      <p class="hint">
        Hint: you can search by multiple keywords, separated by space, e.g.{' '}
        <code>react event</code> as searching for repos containing{' '}
        <code>react</code> and <code>event</code> in the name (appearing any
        order).
      </p>
      <p class="hint">
        Hint: you can indicate negative keywords with hyphen prefix, e.g.{' '}
        <code>-react -ng- chart</code> as searching for <code>chart</code>{' '}
        libraries while excluding those framework-specific libraries having{' '}
        <code>react</code> or <code>ng-</code> in the name.
      </p>
      <p class="hint">
        Hint: multiple keywords are combined with "and" for most fields, but
        they're combined with "or" for programming languages.
      </p>
      <p class="hint">
        Hint: the keyboards are matched partially for most fields, but is
        matched exactly for programming languages. So searching{' '}
        <code>Java</code> will not match <code>Javascript</code> repos.
      </p>
      <p class="hint">
        Hint: if a keyword is wrapped with double quotes, it is matched in full.
        For example searching <code>speed</code> will matched for{' '}
        <code>frank-dspeed</code> but searching <code>"speed"</code> will not
        match for that user. This feature does not apply to the language field
        as it's always matched in full.
      </p>
      {result}
    </form>
  )
}

function MatchedItem(res: MatchedItem) {
  let { desc, programming_language } = res
  return (
    <div class="res">
      <div>
        {ProgrammingLanguageSpan(programming_language)}
        <b>{res.name}</b> {res.deprecated ? <span>(deprecated)</span> : null}{' '}
        {res.username ? <sub>by {res.username}</sub> : null}{' '}
        {res.is_fork ? <sub>(fork)</sub> : null}{' '}
      </div>
      <a target="_blank" href={res.url}>
        {res.url}
      </a>
      {desc ? <div class="res-desc">{desc}</div> : null}
    </div>
  )
}

function build_search_query_test() {
  let schema = db
    .prepare<void[], { name: string; sql: string }>(
      /* sql */ `
select name, sql from sqlite_master
where type = 'table'
  and name not like 'knex%'
  and name <> 'sqlite_sequence'
`,
    )
    .all()

  let testDB = newDB({
    memory: true,
    migrate: false,
  })
  for (let row of schema) {
    testDB.prepare(row.sql).run()
  }

  function seedTable(table: string, field: string, value: string) {
    let id = testDB.queryFirstCell(
      `select id from ${table} where ${field} = ?`,
      value,
    )
    if (!id) {
      id = testDB.insert(table, { [field]: value })
    }
    return id
  }

  function seedRepo(id: number, url: string, programming_language: string) {
    // e.g. [ 'https:', '', 'github.com', 'beenotung', 'create-ts-liveview' ]
    let parts = url.split('/')
    let host = parts[2]
    let username = parts[3]
    let name = parts[4]
    testDB.insert('page', {
      id,
      url,
    })
    testDB.insert('repo', {
      id,
      page_id: id,
      domain_id: seedTable('domain', 'host', host),
      author_id: seedTable('author', 'username', username),
      programming_language_id: seedTable(
        'programming_language',
        'name',
        programming_language,
      ),
      is_public: 1,
      name,
      url,
    })
    return {
      name,
      desc: null,
      url,
      programming_language,
      username,
      is_fork: null,
      deprecated: null,
    }
  }
  let samples = [
    seedRepo(
      1,
      'https://github.com/beenotung/create-ts-liveview',
      'Javascript',
    ),
    seedRepo(2, 'https://github.com/beenotung/ts-liveview', 'Typescript'),
    seedRepo(
      3,
      'https://github.com/beenotung/better-sqlite3-proxy',
      'Typescript',
    ),
    seedRepo(4, 'https://github.com/beenotung/net-files', 'HTML'),
    seedRepo(5, 'https://github.com/beenotung/safepic', 'HTML'),
    seedRepo(6, 'https://github.com/beenotung/ga-experiment', 'Java'),
    seedRepo(7, 'https://github.com/beenotung/vue-datepicker', 'Vue'),
    seedRepo(8, 'https://github.com/beenotung/sodoku', 'C'),
    seedRepo(9, 'https://github.com/beenotung/fair-task-pool', 'Typescript'),
    seedRepo(10, 'https://github.com/valor-software/ng2-charts', 'Typescript'),
    seedRepo(11, 'https://github.com/help-me-mom/ng-mocks', 'Typescript'),
    seedRepo(12, 'https://github.com/DethAriel/ng-recaptcha', 'Typescript'),
  ]

  function testLanguage(name: string, language: string, expected: any[]) {
    name = `[Language TestSuit] ${name}`
    let params = new URLSearchParams({ language })
    test(name, params, expected)
  }
  function testName(name: string, repoName: string, expected: any[]) {
    name = `[Name TestSuit] ${name}`
    let params = new URLSearchParams({ name: repoName })
    test(name, params, expected)
  }
  function test(name: string, params: URLSearchParams, expected: any[]) {
    let query = build_search_query(params)

    let actual = testDB
      .prepare(query.search_repo_sql)
      .all(query.search_repo_bindings)

    if (JSON.stringify(actual) !== JSON.stringify(expected)) {
      console.error('[fail]', name, {
        expected,
        actual,
      })
      process.exit(1)
    }
    console.log('[pass]', name)
  }
  testLanguage('empty query', '', samples)
  testLanguage(
    'single positive language',
    'Typescript',
    samples.filter(repo => repo.programming_language == 'Typescript'),
  )
  testLanguage(
    'multiple positive languages',
    'Typescript Javascript',
    samples.filter(
      repo =>
        repo.programming_language == 'Typescript' ||
        repo.programming_language == 'Javascript',
    ),
  )
  testLanguage(
    'single negative language',
    '-Javascript',
    samples.filter(repo => repo.programming_language != 'Javascript'),
  )
  testLanguage(
    'multiple negative languages',
    '-Typescript -Javascript -Java',
    samples.filter(
      repo =>
        repo.programming_language != 'Typescript' &&
        repo.programming_language != 'Javascript' &&
        repo.programming_language != 'Java',
    ),
  )
  testLanguage(
    'mixed positive and negative languages',
    'Typescript -Javascript',
    samples.filter(repo => repo.programming_language == 'Typescript'),
  )
  testName(
    'single keyword',
    'ga',
    samples.filter(repo => repo.name.includes('ga')),
  )
  testName(
    'multiple keywords',
    'net files',
    samples.filter(
      repo => repo.name.includes('net') && repo.name.includes('files'),
    ),
  )
  testName(
    'negative keyword',
    '-ng',
    samples.filter(repo => !repo.name.includes('ng')),
  )
  testName(
    'positive keyword with hyphen suffix',
    'ng-',
    samples.filter(repo => repo.name.includes('ng-')),
  )
  testName(
    'negative keyword with hyphen suffix',
    '-ng-',
    samples.filter(repo => !repo.name.includes('ng-')),
  )
  testName(
    'multiple keywords with hyphen suffix (continue)',
    'ng- mock',
    samples.filter(
      repo => repo.name.includes('ng-') && repo.name.includes('mock'),
    ),
  )
  testName(
    'multiple keywords with hyphen suffix (separated)',
    'fair- pool',
    samples.filter(
      repo => repo.name.includes('fair-') && repo.name.includes('pool'),
    ),
  )
  console.log('all passed')
}
if (process.argv[1] == import.meta.filename) {
  build_search_query_test()
}

// And it can be pre-rendered into html as well
// let Home = prerender(content)

type SearchContext = DynamicContext & {
  params: URLSearchParams
  query: ReturnType<typeof build_search_query>
}

let content = (
  <div id="home">
    {style}
    <h1>FOSS Git Repository & NPM Package Index</h1>
    <Page />
    {script}
    <SourceCode page="home.tsx" />
  </div>
)

let routes = {
  '/': {
    menuText: 'Search',
    resolve(context) {
      let params = new URLSearchParams(context.routerMatch?.search)
      let query = build_search_query(params)

      let ctx = context as SearchContext
      ctx.params = params
      ctx.query = query

      function getTitle() {
        let acc = ''

        function add(text: string | null): void {
          if (!text) return
          if (acc) {
            acc += ' '
          }
          acc += text
        }

        add(query.desc)
        add(query.name || (acc ? 'resources' : 'Resources'))
        if (query.prefix) {
          add(`(${query.prefix}*)`)
        }
        if (query.language) {
          add('in ' + query.language)
        }
        if (query.host) {
          add('on ' + query.host)
        }
        if (query.username) {
          add('by ' + query.username)
        }

        return acc.trim()
      }

      return {
        title: getTitle() + ' | FOSS Git Repositories & NPM Packages',
        description:
          'Getting Started with ts-liveview - a server-side rendering realtime webapp framework with progressive enhancement',
        node: content,
      }
    },
  },
} satisfies Routes

export default { routes }