Files
supabase/apps/docs/scripts/search/sources/lint-warnings-guide.ts

142 lines
4.0 KiB
TypeScript

import { createAppAuth } from '@octokit/auth-app'
import { Octokit } from '@octokit/core'
import { retry } from '@octokit/plugin-retry'
import crypto, { createHash } from 'node:crypto'
import { OCTOKIT_RETRY_OPTIONS } from '../../../lib/octokit.constants.js'
import { BaseLoader, BaseSource } from './base.js'
const RetryOctokit = Octokit.plugin(retry)
const appId = process.env.DOCS_GITHUB_APP_ID
const installationId = process.env.DOCS_GITHUB_APP_INSTALLATION_ID
const privateKey = process.env.DOCS_GITHUB_APP_PRIVATE_KEY
const getBasename = (path: string) => path.split('/').at(-1)!.replace(/\.md$/, '')
export class LintWarningsGuideLoader extends BaseLoader {
type = 'markdown' as const
constructor(
source: string,
path: string,
public org: string,
public repo: string,
public branch: string,
public docsDir: string
) {
super(source, path)
}
async load() {
if (!appId || !installationId || !privateKey) {
throw new Error('Missing DOCS_GITHUB_APP_* environment variables')
}
const octokit = new RetryOctokit({
authStrategy: createAppAuth,
auth: {
appId,
installationId,
privateKey: crypto.createPrivateKey(privateKey).export({ type: 'pkcs8', format: 'pem' }),
},
})
const response = await octokit.request('GET /repos/{owner}/{repo}/contents/{path}', {
owner: this.org,
repo: this.repo,
path: this.docsDir,
ref: this.branch,
request: OCTOKIT_RETRY_OPTIONS,
headers: {
'X-GitHub-Api-Version': '2022-11-28',
},
})
if (response.status >= 400) {
throw Error(`Could not get contents of repo ${this.org}/${this.repo}`)
}
if (!Array.isArray(response.data)) {
throw Error(
'Reading a directory, not a file. Should not reach this, solely to appease Typescript.'
)
}
const lintsList = response.data.filter(({ path }) => /docs\/\d+.+\.md$/.test(path))
// Fetch all lint files and combine them into a single guide
const lints = await Promise.all(
lintsList.map(async ({ path }) => {
const fileResponse = await octokit.request('GET /repos/{owner}/{repo}/contents/{path}', {
owner: this.org,
repo: this.repo,
path,
ref: this.branch,
request: OCTOKIT_RETRY_OPTIONS,
})
if (!('content' in fileResponse.data) || fileResponse.data.type !== 'file') {
throw Error(`Could not get contents of file ${this.org}/${this.repo}/${path}`)
}
const content = Buffer.from(fileResponse.data.content, 'base64').toString('utf-8')
const basename = getBasename(path)
return {
path: basename,
content,
originalPath: path,
}
})
)
// Create a separate source for each lint file
return lints.map(
(lint) =>
new LintWarningsGuideSource(
this.source,
`${this.path}?queryGroups=lint&lint=${lint.path}`,
lint
)
)
}
}
export class LintWarningsGuideSource extends BaseSource {
type = 'markdown' as const
constructor(
source: string,
path: string,
public lint: {
path: string
content: string
originalPath: string
}
) {
super(source, path)
}
async process() {
this.checksum = createHash('sha256').update(this.lint.content).digest('base64')
this.meta = {
title: `Database Advisor: Lint ${this.lint.path}`,
}
this.sections = [
{
content: this.lint.content,
},
]
return { checksum: this.checksum, meta: this.meta, sections: this.sections }
}
extractIndexedContent(): string {
const sections = this.sections ?? []
const sectionText = sections.map(({ content }) => content).join('\n\n')
return `# Database Advisor: Lint ${this.lint.path}\n\nThis is a database lint rule for Supabase, targeting the lint ID ${this.lint.path}. Lint rules help enforce performance and security best practices for your Supabase database.\n\n${sectionText}`
}
}