mirror of
https://gitea.toothfairyai.com/ToothFairyAI/tf_code.git
synced 2026-03-30 22:03:58 +00:00
129 lines
3.4 KiB
TypeScript
129 lines
3.4 KiB
TypeScript
import { Bus } from "../bus"
|
|
import { File } from "../file"
|
|
import { Log } from "../util/log"
|
|
import path from "path"
|
|
import z from "zod"
|
|
|
|
import * as Formatter from "./formatter"
|
|
import { Config } from "../config/config"
|
|
import { mergeDeep } from "remeda"
|
|
import { Instance } from "../project/instance"
|
|
|
|
export namespace Format {
|
|
const log = Log.create({ service: "format" })
|
|
|
|
export const Status = z
|
|
.object({
|
|
name: z.string(),
|
|
extensions: z.string().array(),
|
|
enabled: z.boolean(),
|
|
})
|
|
.meta({
|
|
ref: "FormatterStatus",
|
|
})
|
|
export type Status = z.infer<typeof Status>
|
|
|
|
const state = Instance.state(async () => {
|
|
const enabled: Record<string, boolean> = {}
|
|
const cfg = await Config.get()
|
|
|
|
const formatters: Record<string, Formatter.Info> = {}
|
|
for (const item of Object.values(Formatter)) {
|
|
formatters[item.name] = item
|
|
}
|
|
for (const [name, item] of Object.entries(cfg.formatter ?? {})) {
|
|
if (item.disabled) {
|
|
delete formatters[name]
|
|
continue
|
|
}
|
|
const result: Formatter.Info = mergeDeep(formatters[name] ?? {}, {
|
|
command: [],
|
|
extensions: [],
|
|
...item,
|
|
})
|
|
result.enabled = async () => true
|
|
result.name = name
|
|
formatters[name] = result
|
|
}
|
|
|
|
return {
|
|
enabled,
|
|
formatters,
|
|
}
|
|
})
|
|
|
|
async function isEnabled(item: Formatter.Info) {
|
|
const s = await state()
|
|
let status = s.enabled[item.name]
|
|
if (status === undefined) {
|
|
status = await item.enabled()
|
|
s.enabled[item.name] = status
|
|
}
|
|
return status
|
|
}
|
|
|
|
async function getFormatter(ext: string) {
|
|
const formatters = await state().then((x) => x.formatters)
|
|
const result = []
|
|
for (const item of Object.values(formatters)) {
|
|
log.info("checking", { name: item.name, ext })
|
|
if (!item.extensions.includes(ext)) continue
|
|
if (!(await isEnabled(item))) continue
|
|
log.info("enabled", { name: item.name, ext })
|
|
result.push(item)
|
|
}
|
|
return result
|
|
}
|
|
|
|
export async function status() {
|
|
const s = await state()
|
|
const result: Status[] = []
|
|
for (const formatter of Object.values(s.formatters)) {
|
|
const enabled = await isEnabled(formatter)
|
|
result.push({
|
|
name: formatter.name,
|
|
extensions: formatter.extensions,
|
|
enabled,
|
|
})
|
|
}
|
|
return result
|
|
}
|
|
|
|
export function init() {
|
|
log.info("init")
|
|
return
|
|
Bus.subscribe(File.Event.Edited, async (payload) => {
|
|
const file = payload.properties.file
|
|
log.info("formatting", { file })
|
|
const ext = path.extname(file)
|
|
|
|
for (const item of await getFormatter(ext)) {
|
|
log.info("running", { command: item.command })
|
|
try {
|
|
const proc = Bun.spawn({
|
|
cmd: item.command.map((x) => x.replace("$FILE", file)),
|
|
cwd: Instance.directory,
|
|
env: { ...process.env, ...item.environment },
|
|
stdout: "ignore",
|
|
stderr: "ignore",
|
|
})
|
|
const exit = await proc.exited
|
|
if (exit !== 0)
|
|
log.error("failed", {
|
|
command: item.command,
|
|
...item.environment,
|
|
})
|
|
} catch (error) {
|
|
log.error("failed", {
|
|
error,
|
|
command: item.command,
|
|
...item.environment,
|
|
})
|
|
// re-raising
|
|
throw error
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|