feat(core): 初始化项目基础架构和CMS功能模块

- 添加Docker相关配置文件(.dockerignore, .env.example, .gitignore)
- 实现服务端API代理功能,支持文件、模块和服务器API转发
- 创建文章详情页、栏目文章列表页和单页内容展示页面
- 集成Ant Design Vue组件库并实现SSR样式提取功能
- 定义API响应数据结构类型和应用布局组件
- 开发开发者应用中心和文章管理页面
- 实现CMS导航菜单获取和多租户切换功能
This commit is contained in:
2026-01-27 00:14:08 +08:00
commit 775841eed3
315 changed files with 47072 additions and 0 deletions

View File

@@ -0,0 +1,25 @@
import { defineEventHandler, getHeader, getRequestURL, getRouterParam, proxyRequest } from 'h3'
import { useRuntimeConfig } from '#imports'
function joinURL(base: string, path: string) {
if (!path) return base
return base.replace(/\/+$/, '') + '/' + path.replace(/^\/+/, '')
}
export default defineEventHandler((event) => {
const config = useRuntimeConfig()
const fileServerBase = config.public.fileServerBase || 'https://server.websoft.top'
const path = getRouterParam(event, 'path') || ''
const search = getRequestURL(event).search
const target = joinURL(fileServerBase, path) + search
const tenantId = getHeader(event, 'tenantid') || config.public.tenantId
const authorization = getHeader(event, 'authorization')
return proxyRequest(event, target, {
headers: {
TenantId: String(tenantId),
...(authorization ? { Authorization: String(authorization) } : {})
}
})
})

View File

@@ -0,0 +1,26 @@
import { defineEventHandler, getHeader, getRequestURL, getRouterParam, proxyRequest } from 'h3'
import { useRuntimeConfig } from '#imports'
function joinURL(base: string, path: string) {
if (!path) return base
return base.replace(/\/+$/, '') + '/' + path.replace(/^\/+/, '')
}
export default defineEventHandler((event) => {
const config = useRuntimeConfig()
const modulesApiBase =
config.public.modulesApiBase || config.public.ApiBase || 'https://cms-api.websoft.top/api'
const path = getRouterParam(event, 'path') || ''
const search = getRequestURL(event).search
const target = joinURL(modulesApiBase, path) + search
const tenantId = getHeader(event, 'tenantid') || config.public.tenantId
const authorization = getHeader(event, 'authorization')
return proxyRequest(event, target, {
headers: {
TenantId: String(tenantId),
...(authorization ? { Authorization: String(authorization) } : {})
}
})
})

View File

@@ -0,0 +1,25 @@
import { defineEventHandler, getHeader, getRequestURL, getRouterParam, proxyRequest } from 'h3'
import { useRuntimeConfig } from '#imports'
function joinURL(base: string, path: string) {
if (!path) return base
return base.replace(/\/+$/, '') + '/' + path.replace(/^\/+/, '')
}
export default defineEventHandler((event) => {
const config = useRuntimeConfig()
const serverApiBase = config.public.serverApiBase || config.public.ServerApi || 'https://server.websoft.top/api'
const path = getRouterParam(event, 'path') || ''
const search = getRequestURL(event).search
const target = joinURL(serverApiBase, path) + search
const tenantId = getHeader(event, 'tenantid') || config.public.tenantId
const authorization = getHeader(event, 'authorization')
return proxyRequest(event, target, {
headers: {
TenantId: String(tenantId),
...(authorization ? { Authorization: String(authorization) } : {})
}
})
})

View File

@@ -0,0 +1,42 @@
import { $fetch } from 'ofetch'
import { createError, defineEventHandler, getHeader, getQuery } from 'h3'
import { useRuntimeConfig } from '#imports'
// Frontend-friendly endpoint:
// GET /api/cms-navigation?parentId=xxxx
// Proxies to CMS modules API. Some deployments expose "/cms-navigation", others "/cms/cms-navigation".
export default defineEventHandler(async (event) => {
const config = useRuntimeConfig()
const query = getQuery(event)
const modulesApiBase =
config.public.modulesApiBase || config.public.ApiBase || 'https://cms-api.websoft.top/api'
const tenantId =
getHeader(event, 'tenantid') ||
config.public.tenantId ||
config.public.TenantId ||
'10584'
const authorization = getHeader(event, 'authorization')
const headers = {
TenantId: String(tenantId),
...(authorization ? { Authorization: String(authorization) } : {})
}
const upstreamCandidates = ['/cms-navigation', '/cms/cms-navigation']
let lastError: any
for (const path of upstreamCandidates) {
try {
return await $fetch(path, { baseURL: modulesApiBase, headers, query })
} catch (error: any) {
lastError = error
}
}
throw createError({
statusCode: lastError?.statusCode || lastError?.response?.status || 502,
statusMessage: lastError?.statusMessage || 'Failed to fetch cms navigation'
})
})

View File

@@ -0,0 +1,2 @@
export { default } from '../cms-navigation.get'

View File

@@ -0,0 +1,33 @@
import { $fetch } from 'ofetch'
import { createError, defineEventHandler, getHeader, getQuery } from 'h3'
import { useRuntimeConfig } from '#imports'
export default defineEventHandler(async (event) => {
const config = useRuntimeConfig()
const query = getQuery(event)
const modulesApiBase =
config.public.modulesApiBase || config.public.ApiBase || 'https://cms-api.websoft.top/api'
const tenantId =
getHeader(event, 'tenantid') ||
config.public.tenantId ||
config.public.TenantId ||
'10584'
const authorization = getHeader(event, 'authorization')
try {
return await $fetch('/cms/cms-website/getSiteInfo', {
baseURL: modulesApiBase,
headers: {
TenantId: String(tenantId),
...(authorization ? { Authorization: String(authorization) } : {})
},
query
})
} catch (error: any) {
throw createError({
statusCode: error?.statusCode || error?.response?.status || 502,
statusMessage: error?.statusMessage || 'Failed to fetch site info'
})
}
})