mirror of
https://github.com/supabase/supabase.git
synced 2026-05-11 02:20:29 +08:00
142 lines
4.0 KiB
TypeScript
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}`
|
|
}
|
|
}
|