Skip to content

Commit

Permalink
Detect github release tag format via API calls (#116)
Browse files Browse the repository at this point in the history
RELATED #58
  • Loading branch information
hobofan authored Apr 28, 2024
1 parent c512660 commit 32f3dcb
Showing 1 changed file with 125 additions and 7 deletions.
132 changes: 125 additions & 7 deletions pages/modules/[module].tsx
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'
import { faGithub } from '@fortawesome/free-brands-svg-icons'
import { faEnvelope } from '@fortawesome/free-regular-svg-icons'
import { CopyCode } from '../../components/CopyCode'
import React, { useState } from 'react'
import React, { useEffect, useState } from 'react'
import {
getStaticPropsModulePage,
VersionInfo,
Expand All @@ -36,6 +36,11 @@ const ModulePage: NextPage<ModulePageProps> = ({
}) => {
const router = useRouter()
const { module } = router.query
// There may be multiple GitHub repositories specified for a module, but for now
// the UI will only display information about the first one in the list.
const firstGithubRepository = metadata.repository?.find((repo) =>
repo.startsWith('github:')
)

const [triggeredShowAllVersions, setTriggeredShowAllVersions] =
useState(false)
Expand All @@ -44,6 +49,11 @@ const ModulePage: NextPage<ModulePageProps> = ({
setTriggeredShowAllReverseDependencies,
] = useState(false)

const releaseTagFormat = useDetectReleaseFormatViaGithubApi(
firstGithubRepository,
selectedVersion
)

const isQualifiedForShowAllVersions =
versionInfos.length > NUM_VERSIONS_ON_PAGE_LOAD
const displayShowAllVersionsButton =
Expand All @@ -63,12 +73,15 @@ const ModulePage: NextPage<ModulePageProps> = ({

const versionInfo = versionInfos.find((n) => n.version === selectedVersion)

const githubLink = metadata.repository
?.find((repo) => repo.startsWith('github:'))
?.replace('github:', 'https://github.com/')
const releaseNotesLink = githubLink
? `${githubLink}/releases/tag/v${selectedVersion}`
: undefined
const githubLink = firstGithubRepository?.replace(
'github:',
'https://github.com/'
)
const releaseNotesLink = buildReleaseNotesLink(
githubLink,
selectedVersion,
releaseTagFormat
)

if (!versionInfo) {
throw Error(
Expand Down Expand Up @@ -393,4 +406,109 @@ export async function getStaticPaths() {
}
}

/**
* Encodes the format that release tags on the GitHub repository have.
*
* Allows us to build correct links to the release notes.
*/
enum ReleaseTagFormat {
/**
* Tag `1.2.3` for the version number `1.2.3`.
*/
NO_PREFIX,
/**
* Tag `v1.2.3` for the version number `1.2.3`.
*/
V_PREFIX,
/**
* Other tag format. Can occur if our heuristics can't detect a concrete format.
*/
UNKNOWN,
}

const buildReleaseNotesLink = (
githubLink: string | undefined,
moduleVersion: string,
releaseTagFormat: ReleaseTagFormat
): string | undefined => {
if (!githubLink) {
return undefined
}

switch (releaseTagFormat) {
case ReleaseTagFormat.NO_PREFIX:
return `${githubLink}/releases/tag/${moduleVersion}`
case ReleaseTagFormat.V_PREFIX:
return `${githubLink}/releases/tag/v${moduleVersion}`
case ReleaseTagFormat.UNKNOWN:
// If we don't know to format, we'll link to the release search for that module version.
// For many cases (typo in repo, multiple modules per repo), this is still more desirable than a 404.
return `${githubLink}/releases?q=${moduleVersion}`
}
// @ts-ignore: Unreachable code error
throw new Error(
'Unable to generate release notes link due to unknown release tag format. Should be unreachable.'
)
}

type UseDetectReleaseFormatViaGithubApiReturns = ReleaseTagFormat

/**
* Hook that detects the applicable `ReleaseTagFormat` for a module by sending 1-2 requests to the Github API from the browser.
*/
const useDetectReleaseFormatViaGithubApi = (
metadataRepository: string | undefined,
moduleVersion: string
): UseDetectReleaseFormatViaGithubApiReturns => {
const githubOwnerAndRepo = metadataRepository?.replace('github:', '')
// We default to `UNKNOWN`, so that we still have a reasonable default in case that e.g. the API request fails.
const [releaseTagFormat, setReleaseTagFormat] = useState(
ReleaseTagFormat.UNKNOWN
)
useEffect(() => {
const detectReleaseFormat = async () => {
// Don't send any requests if we don't have a repo.
if (!githubOwnerAndRepo) {
return
}
// First try is with v-prefix, as that is the most common.
const vPrefixResponse = await fetch(
`https://api.github.com/repos/${githubOwnerAndRepo}/releases/tags/v${moduleVersion}`,
{
method: 'GET',
headers: {
Accept: 'application/vnd.github+json',
'User-Agent': 'Bazel Central Registry UI',
'X-GitHub-Api-Version': '2022-11-28',
},
}
)
if (vPrefixResponse.ok) {
setReleaseTagFormat(ReleaseTagFormat.V_PREFIX)
return
}
// Second try without prefix
const noPrefixResponse = await fetch(
`https://api.github.com/repos/${githubOwnerAndRepo}/releases/tags/${moduleVersion}`,
{
method: 'GET',
headers: {
Accept: 'application/vnd.github+json',
'User-Agent': 'Bazel Central Registry UI',
'X-GitHub-Api-Version': '2022-11-28',
},
}
)
if (noPrefixResponse.ok) {
setReleaseTagFormat(ReleaseTagFormat.NO_PREFIX)
return
}
// Neither matches -> Leave format as default value.
}
detectReleaseFormat()
}, [githubOwnerAndRepo, moduleVersion])

return releaseTagFormat
}

export default ModulePage

0 comments on commit 32f3dcb

Please sign in to comment.