forked from imbytecat/fullstack-starter
- 添加 Rust、Tauri 和 Even Better TOML 语言支持扩展以增强开发环境功能。 - 修改构建配置中的输出目录为 src-tauri/binaries,以适配新的构建输出路径。 - 添加 Tauri CLI 及其各平台兼容的可执行文件依赖,以支持多平台应用开发。 - 添加最新版本的 Rust 工具支持并移除旧版本的 Bun。 - 添加 Tauri CLI 工具以支持桌面应用开发 - 添加 Tauri 项目生成文件和构建产物的忽略规则 - 新增项目开发指南文档,明确 Tauri 桌面应用的构建流程、代码风格、Tauri 特定规范及最佳实践。 - 添加 Tauri 构建脚本以支持应用打包和构建流程 - 添加主窗口权限配置,允许执行名为 binaries/server 的侧车二进制文件。 - 生成新的 Cargo.lock 文件以更新项目依赖项的版本和校验和 - 添加 Tauri 应用的 Cargo 项目配置,包含 shell 插件和异步运行时支持。 - 添加32x32像素图标文件以支持应用图标显示 - 添加128x128像素图标文件以支持应用图标显示 - 添加高分辨率图标文件以支持高清显示 - 添加应用图标文件以支持 macOS 平台的图标显示 - 添加图标文件以用于应用程序的视觉标识 - 添加应用图标文件 - 添加方形30x30像素的图标文件以支持应用启动画面和系统显示 - 添加方形44x44像素应用图标文件 - 添加方形71x71像素应用图标以支持不同平台显示需求 - 添加新的方形89x89像素应用图标文件 - 添加新的正方形107x107像素应用图标文件 - 添加方形142x142像素的图标文件以支持应用启动画面和系统显示 - 添加新的方形150x150像素图标文件以用于应用程序界面显示 - 添加新的方形284x284像素应用图标文件 - 添加新的方形310x310像素应用图标文件 - 添加应用商店图标文件 - 添加原生桌面功能命令模块,包含示例问候函数以支持从 Rust 向前端传递消息。 - 添加侧车进程管理功能并注册全局状态与生命周期事件处理 - 添加启动配置以在发布模式下防止Windows出现额外控制台窗口,并启动应用主函数。 - 添加 Sidecar 服务启动与管理功能,自动查找可用端口、启动后端服务并创建主窗口,同时支持超时检测和进程清理。 - 添加 Tauri 项目配置文件并设置应用基本信息、打包选项及图标路径
290 lines
7.9 KiB
TypeScript
290 lines
7.9 KiB
TypeScript
import { Schema } from '@effect/schema'
|
|
import { $ } from 'bun'
|
|
import { Console, Context, Data, Effect, Layer } from 'effect'
|
|
|
|
// ============================================================================
|
|
// Domain Models & Schema
|
|
// ============================================================================
|
|
|
|
const targetMap = {
|
|
'bun-windows-x64': 'x86_64-pc-windows-msvc',
|
|
'bun-darwin-arm64': 'aarch64-apple-darwin',
|
|
'bun-darwin-x64': 'x86_64-apple-darwin',
|
|
'bun-linux-x64': 'x86_64-unknown-linux-gnu',
|
|
'bun-linux-arm64': 'aarch64-unknown-linux-gnu',
|
|
} as const
|
|
|
|
const BunTargetSchema = Schema.Literal(
|
|
'bun-windows-x64',
|
|
'bun-darwin-arm64',
|
|
'bun-darwin-x64',
|
|
'bun-linux-x64',
|
|
'bun-linux-arm64',
|
|
)
|
|
|
|
type BunTarget = Schema.Schema.Type<typeof BunTargetSchema>
|
|
|
|
const BuildConfigSchema = Schema.Struct({
|
|
entrypoint: Schema.String.pipe(Schema.nonEmptyString()),
|
|
outputDir: Schema.String.pipe(Schema.nonEmptyString()),
|
|
targets: Schema.Array(BunTargetSchema).pipe(Schema.minItems(1)),
|
|
})
|
|
|
|
type BuildConfig = Schema.Schema.Type<typeof BuildConfigSchema>
|
|
|
|
const BuildResultSchema = Schema.Struct({
|
|
target: BunTargetSchema,
|
|
outputs: Schema.Array(Schema.String),
|
|
})
|
|
|
|
type BuildResult = Schema.Schema.Type<typeof BuildResultSchema>
|
|
|
|
// ============================================================================
|
|
// Error Models (使用 Data.TaggedError)
|
|
// ============================================================================
|
|
|
|
class CleanError extends Data.TaggedError('CleanError')<{
|
|
readonly dir: string
|
|
readonly cause: unknown
|
|
}> {}
|
|
|
|
class BuildError extends Data.TaggedError('BuildError')<{
|
|
readonly target: BunTarget
|
|
readonly cause: unknown
|
|
}> {}
|
|
|
|
class ConfigError extends Data.TaggedError('ConfigError')<{
|
|
readonly message: string
|
|
readonly cause: unknown
|
|
}> {}
|
|
|
|
// ============================================================================
|
|
// Services
|
|
// ============================================================================
|
|
|
|
/**
|
|
* 配置服务
|
|
*/
|
|
class BuildConfigService extends Context.Tag('BuildConfigService')<
|
|
BuildConfigService,
|
|
BuildConfig
|
|
>() {
|
|
/**
|
|
* 从原始数据创建并验证配置
|
|
*/
|
|
static fromRaw = (raw: unknown) =>
|
|
Effect.gen(function* () {
|
|
const decoded = yield* Schema.decodeUnknown(BuildConfigSchema)(raw)
|
|
return decoded
|
|
}).pipe(
|
|
Effect.catchAll((error) =>
|
|
Effect.fail(
|
|
new ConfigError({
|
|
message: '配置验证失败',
|
|
cause: error,
|
|
}),
|
|
),
|
|
),
|
|
)
|
|
|
|
/**
|
|
* 默认配置 Layer
|
|
*/
|
|
static readonly Live = Layer.effect(
|
|
BuildConfigService,
|
|
BuildConfigService.fromRaw({
|
|
entrypoint: './.output/server/index.mjs',
|
|
// outputDir: './out',
|
|
outputDir: './src-tauri/binaries',
|
|
targets: ['bun-windows-x64', 'bun-darwin-arm64', 'bun-linux-x64'],
|
|
}),
|
|
)
|
|
}
|
|
|
|
/**
|
|
* 文件系统服务
|
|
*/
|
|
class FileSystemService extends Context.Tag('FileSystemService')<
|
|
FileSystemService,
|
|
{
|
|
readonly cleanDir: (dir: string) => Effect.Effect<void, CleanError>
|
|
}
|
|
>() {
|
|
static readonly Live = Layer.succeed(FileSystemService, {
|
|
cleanDir: (dir: string) =>
|
|
Effect.tryPromise({
|
|
try: async () => {
|
|
await $`rm -rf ${dir}`
|
|
},
|
|
catch: (cause: unknown) =>
|
|
new CleanError({
|
|
dir,
|
|
cause,
|
|
}),
|
|
}),
|
|
})
|
|
}
|
|
|
|
/**
|
|
* 构建服务
|
|
*/
|
|
class BuildService extends Context.Tag('BuildService')<
|
|
BuildService,
|
|
{
|
|
readonly buildForTarget: (
|
|
config: BuildConfig,
|
|
target: BunTarget,
|
|
) => Effect.Effect<BuildResult, BuildError>
|
|
readonly buildAll: (
|
|
config: BuildConfig,
|
|
) => Effect.Effect<ReadonlyArray<BuildResult>, BuildError>
|
|
}
|
|
>() {
|
|
static readonly Live = Layer.succeed(BuildService, {
|
|
buildForTarget: (config: BuildConfig, target: BunTarget) =>
|
|
Effect.gen(function* () {
|
|
yield* Console.log(`🔨 开始构建: ${target}`)
|
|
|
|
const output = yield* Effect.tryPromise({
|
|
try: () =>
|
|
Bun.build({
|
|
entrypoints: [config.entrypoint],
|
|
compile: {
|
|
outfile: `server-${targetMap[target]}`,
|
|
target: target,
|
|
},
|
|
outdir: config.outputDir,
|
|
}),
|
|
catch: (cause: unknown) =>
|
|
new BuildError({
|
|
target,
|
|
cause,
|
|
}),
|
|
})
|
|
|
|
const paths = output.outputs.map((item: { path: string }) => item.path)
|
|
|
|
return {
|
|
target,
|
|
outputs: paths,
|
|
} satisfies BuildResult
|
|
}),
|
|
|
|
buildAll: (config: BuildConfig) =>
|
|
Effect.gen(function* () {
|
|
const effects = config.targets.map((target) =>
|
|
Effect.gen(function* () {
|
|
yield* Console.log(`🔨 开始构建: ${target}`)
|
|
|
|
const output = yield* Effect.tryPromise({
|
|
try: () =>
|
|
Bun.build({
|
|
entrypoints: [config.entrypoint],
|
|
compile: {
|
|
outfile: `server-${targetMap[target]}`,
|
|
target: target,
|
|
},
|
|
outdir: config.outputDir,
|
|
}),
|
|
catch: (cause: unknown) =>
|
|
new BuildError({
|
|
target,
|
|
cause,
|
|
}),
|
|
})
|
|
|
|
const paths = output.outputs.map(
|
|
(item: { path: string }) => item.path,
|
|
)
|
|
|
|
return {
|
|
target,
|
|
outputs: paths,
|
|
} satisfies BuildResult
|
|
}),
|
|
)
|
|
return yield* Effect.all(effects, { concurrency: 'unbounded' })
|
|
}),
|
|
})
|
|
}
|
|
|
|
/**
|
|
* 报告服务
|
|
*/
|
|
class ReporterService extends Context.Tag('ReporterService')<
|
|
ReporterService,
|
|
{
|
|
readonly printSummary: (
|
|
results: ReadonlyArray<BuildResult>,
|
|
) => Effect.Effect<void>
|
|
}
|
|
>() {
|
|
static readonly Live = Layer.succeed(ReporterService, {
|
|
printSummary: (results: ReadonlyArray<BuildResult>) =>
|
|
Effect.gen(function* () {
|
|
yield* Console.log('\n📦 构建完成:')
|
|
for (const result of results) {
|
|
yield* Console.log(` ${result.target}:`)
|
|
for (const path of result.outputs) {
|
|
yield* Console.log(` - ${path}`)
|
|
}
|
|
}
|
|
}),
|
|
})
|
|
}
|
|
|
|
// ============================================================================
|
|
// Main Program
|
|
// ============================================================================
|
|
|
|
const program = Effect.gen(function* () {
|
|
const config = yield* BuildConfigService
|
|
const fs = yield* FileSystemService
|
|
const builder = yield* BuildService
|
|
const reporter = yield* ReporterService
|
|
|
|
// 1. 清理输出目录
|
|
yield* fs.cleanDir(config.outputDir)
|
|
yield* Console.log(`✓ 已清理输出目录: ${config.outputDir}`)
|
|
|
|
// 2. 并行构建所有目标
|
|
const results = yield* builder.buildAll(config)
|
|
|
|
// 3. 输出构建摘要
|
|
yield* reporter.printSummary(results)
|
|
|
|
return results
|
|
})
|
|
|
|
// ============================================================================
|
|
// Layer Composition
|
|
// ============================================================================
|
|
|
|
const MainLayer = Layer.mergeAll(
|
|
BuildConfigService.Live,
|
|
FileSystemService.Live,
|
|
BuildService.Live,
|
|
ReporterService.Live,
|
|
)
|
|
|
|
// ============================================================================
|
|
// Runner
|
|
// ============================================================================
|
|
|
|
const runnable = program.pipe(
|
|
Effect.provide(MainLayer),
|
|
Effect.catchTags({
|
|
CleanError: (error) =>
|
|
Console.error(`❌ 清理目录失败: ${error.dir}`, error.cause),
|
|
BuildError: (error) =>
|
|
Console.error(`❌ 构建失败 [${error.target}]:`, error.cause),
|
|
ConfigError: (error) =>
|
|
Console.error(`❌ 配置错误: ${error.message}`, error.cause),
|
|
}),
|
|
Effect.tapErrorCause((cause) => Console.error('❌ 未预期的错误:', cause)),
|
|
)
|
|
|
|
Effect.runPromise(runnable).catch(() => {
|
|
process.exit(1)
|
|
})
|