You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
vitepress/lib/server.ts

87 lines
2.4 KiB

import path from 'path'
5 years ago
import { promises as fs } from 'fs'
5 years ago
import {
createServer as createViteServer,
cachedRead,
Plugin,
Resolver
} from 'vite'
5 years ago
import { markdownToVue } from './markdown'
const debug = require('debug')('vitepress')
// built ts files are placed into /dist
5 years ago
const appPath = path.join(__dirname, '../lib/app')
// TODO detect user configured theme
5 years ago
const themePath = path.join(__dirname, '../lib/theme-default')
5 years ago
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\/?/, ''))
}
5 years ago
},
fileToPublic(filePath) {
if (filePath.startsWith(appPath)) {
return `/@app/${path.relative(appPath, filePath)}`
}
if (filePath.startsWith(themePath)) {
return `/@theme/${path.relative(themePath, filePath)}`
}
}
}
5 years ago
const VitePressPlugin: Plugin = ({ app, root, watcher, resolver }) => {
5 years ago
// 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)
}
5 years ago
// 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))
}
})
5 years ago
app.use(async (ctx, next) => {
5 years ago
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()
}
5 years ago
// detect and serve vitepress files
const file = VitePressResolver.publicToFile(ctx.path, root)
if (file) {
ctx.type = path.extname(file)
5 years ago
await cachedRead(ctx, file)
5 years ago
debug(`serving file: ${ctx.url}`)
return next()
}
await next()
5 years ago
// serve our index.html after vite history fallback
if (ctx.url === '/index.html') {
5 years ago
await cachedRead(ctx, path.join(appPath, 'index-dev.html'))
}
})
}
export function createServer() {
return createViteServer({
5 years ago
plugins: [VitePressPlugin],
resolvers: [VitePressResolver]
})
}