import path from 'path' import { promises as fs } from 'fs' import { createServer as createViteServer, cachedRead, Plugin, Resolver } from 'vite' import { markdownToVue } from './markdown' const debug = require('debug')('vitepress') // built ts files are placed into /dist const appPath = path.join(__dirname, '../lib/app') // TODO detect user configured theme const themePath = path.join(__dirname, '../lib/theme-default') const VitePressResolver: Resolver = { publicToFile(publicPath) { if (publicPath.startsWith('/@app')) { return path.join(appPath, publicPath.replace(/^\/@app\/?/, '')) } if (publicPath.startsWith('/@theme')) { return path.join(themePath, publicPath.replace(/^\/@theme\/?/, '')) } }, fileToPublic(filePath) { if (filePath.startsWith(appPath)) { return `/@app/${path.relative(appPath, filePath)}` } if (filePath.startsWith(themePath)) { return `/@theme/${path.relative(themePath, filePath)}` } } } const VitePressPlugin: Plugin = ({ app, root, watcher, resolver }) => { // watch theme files if it's outside of project root if (path.relative(root, themePath).startsWith('..')) { debug(`watching theme dir outside of project root: ${themePath}`) watcher.add(themePath) } // hot reload .md files as .vue files watcher.on('change', async (file) => { if (file.endsWith('.md')) { const content = await fs.readFile(file, 'utf-8') watcher.handleVueReload(file, Date.now(), markdownToVue(content)) } }) app.use(async (ctx, next) => { if (ctx.path.endsWith('.md')) { await cachedRead(ctx, resolver.publicToFile(ctx.path)) // let vite know this is supposed to be treated as vue file ctx.vue = true ctx.body = markdownToVue(ctx.body) debug(`serving ${ctx.url}`) return next() } // detect and serve vitepress files const file = VitePressResolver.publicToFile(ctx.path, root) if (file) { ctx.type = path.extname(file) await cachedRead(ctx, file) debug(`serving file: ${ctx.url}`) return next() } await next() // serve our index.html after vite history fallback if (ctx.url === '/index.html') { await cachedRead(ctx, path.join(appPath, 'index-dev.html')) } }) } export function createServer() { return createViteServer({ plugins: [VitePressPlugin], resolvers: [VitePressResolver] }) }