修改 Dockerfile,解决构建时错误

This commit is contained in:
2025-04-21 13:59:49 +08:00
parent 97b14a5455
commit e905e88d1d
6 changed files with 39 additions and 46 deletions

View File

@@ -1,45 +1,46 @@
# 基础阶段:安装依赖和构建应用 # 基础阶段:安装依赖和构建应用
FROM node:22.14-alpine AS base FROM node:22.14-alpine AS base
# 配置 Alpine 镜像源
# 配置 alpine 镜像源 & 安装依赖
RUN sed -i 's/dl-cdn.alpinelinux.org/mirrors.aliyun.com/g' /etc/apk/repositories RUN sed -i 's/dl-cdn.alpinelinux.org/mirrors.aliyun.com/g' /etc/apk/repositories
RUN apk add cairo pango giflib libjpeg pixman
# 安装 pnpm
RUN corepack enable && corepack prepare pnpm@10.5.2 --activate
# 配置pnpm镜像源
RUN pnpm config set registry https://registry.npmmirror.com
# 安装构建依赖项
RUN apk add --no-cache libc6-compat
WORKDIR /app
# 依赖缓存阶段 # 依赖缓存阶段
FROM base AS deps FROM base AS deps
WORKDIR /app
# 安装依赖
RUN apk add build-base g++ cairo-dev pango-dev giflib-dev
# 启用 pnpm & 配置镜像源
RUN corepack enable pnpm
RUN pnpm config set registry https://registry.npmmirror.com
COPY package.json pnpm-lock.yaml* ./ COPY package.json pnpm-lock.yaml* ./
RUN pnpm fetch --prod RUN pnpm install --frozen-lockfile
RUN pnpm install --frozen-lockfile --prod
# 构建阶段 # 构建阶段
FROM base AS builder FROM base AS builder
WORKDIR /app WORKDIR /app
COPY --from=deps /app/node_modules ./node_modules COPY --from=deps /app/node_modules ./node_modules
COPY . . COPY . .
# 环境变量设置 # 环境变量设置
ENV NEXT_TELEMETRY_DISABLED 1 ENV NEXT_TELEMETRY_DISABLED=1
ENV NODE_ENV production
# 构建应用 # 构建应用
RUN pnpm build RUN corepack enable pnpm
RUN pnpm run build
# 生产阶段 # 生产阶段
FROM node:22.14-alpine AS runner FROM base AS runner
WORKDIR /app WORKDIR /app
# 环境变量设置 # 环境变量设置
ENV NODE_ENV production ENV NODE_ENV=production
ENV NEXT_TELEMETRY_DISABLED 1 ENV NEXT_TELEMETRY_DISABLED=1
# 添加非 root 用户 # 添加非 root 用户
RUN addgroup --system --gid 1001 nodejs RUN addgroup --system --gid 1001 nodejs
@@ -47,12 +48,6 @@ RUN adduser --system --uid 1001 nextjs
# 复制必要文件 # 复制必要文件
COPY --from=builder /app/public ./public COPY --from=builder /app/public ./public
# 设置适当的权限
RUN mkdir .next
RUN chown -R nextjs:nodejs .next
# 复制构建产物
COPY --from=builder --chown=nextjs:nodejs /app/.next/standalone ./ COPY --from=builder --chown=nextjs:nodejs /app/.next/standalone ./
COPY --from=builder --chown=nextjs:nodejs /app/.next/static ./.next/static COPY --from=builder --chown=nextjs:nodejs /app/.next/static ./.next/static
@@ -63,8 +58,8 @@ USER nextjs
EXPOSE 3000 EXPOSE 3000
# 设置环境变量 # 设置环境变量
ENV PORT 3000 ENV PORT=3000
ENV HOSTNAME "0.0.0.0" ENV HOSTNAME="0.0.0.0"
# 启动命令 # 启动命令
CMD ["node", "server.js"] CMD ["node", "server.js"]

View File

@@ -1,18 +1,7 @@
import type { NextConfig } from "next" import type { NextConfig } from "next"
const nextConfig: NextConfig = { const nextConfig: NextConfig = {
poweredByHeader: false,
reactStrictMode: true,
swcMinify: true,
output: "standalone", output: "standalone",
images: {
minimumCacheTTL: 60,
formats: ["image/webp"],
},
experimental: {
optimizeCss: true,
serverComponentsExternalPackages: [],
},
} }
export default nextConfig export default nextConfig

View File

@@ -1,5 +1,5 @@
'use client' 'use client'
import {useEffect, useState} from 'react' import {Suspense, useEffect, useState} from 'react'
import {useSearchParams} from 'next/navigation' import {useSearchParams} from 'next/navigation'
import {IdentifyCallback} from '@/actions/auth/identify' import {IdentifyCallback} from '@/actions/auth/identify'
import {Card, CardContent} from '@/components/ui/card' import {Card, CardContent} from '@/components/ui/card'
@@ -8,6 +8,14 @@ import {CheckCircle, AlertCircle, Loader2} from 'lucide-react'
export type PageProps = {} export type PageProps = {}
export default function Page(props: PageProps) { export default function Page(props: PageProps) {
return (
<Suspense>
<Page1/>
</Suspense>
)
}
function Page1() {
const params = useSearchParams() const params = useSearchParams()
const success = params.get('success') === 'true' const success = params.get('success') === 'true'
const id = params.get('id') || '' const id = params.get('id') || ''

View File

@@ -14,7 +14,7 @@ function generateCaptchaText(length: number = 4): string {
} }
// 哈希验证码文本并使用随机盐值 // 哈希验证码文本并使用随机盐值
function hashCaptcha(text: string): { hash: string; salt: string } { function hashCaptcha(text: string): { hash: string, salt: string } {
const salt = crypto.randomBytes(16).toString('hex') const salt = crypto.randomBytes(16).toString('hex')
const hash = crypto const hash = crypto
.createHmac('sha256', salt) .createHmac('sha256', salt)

View File

@@ -2,6 +2,7 @@ import {useCallback, useEffect, useState} from 'react'
import {Dialog, DialogContent, DialogFooter, DialogHeader, DialogTitle} from '@/components/ui/dialog' import {Dialog, DialogContent, DialogFooter, DialogHeader, DialogTitle} from '@/components/ui/dialog'
import {Button} from '@/components/ui/button' import {Button} from '@/components/ui/button'
import {Input} from '@/components/ui/input' import {Input} from '@/components/ui/input'
import Image from 'next/image'
export type CaptchaProps = { export type CaptchaProps = {
showCaptcha: boolean showCaptcha: boolean
@@ -44,7 +45,7 @@ export default function Captcha(props: CaptchaProps) {
</DialogHeader> </DialogHeader>
<div className="space-y-4"> <div className="space-y-4">
<div className="flex justify-between items-center"> <div className="flex justify-between items-center">
<img <Image
src={captchaImage} src={captchaImage}
alt="验证码" alt="验证码"
width={180} width={180}

View File

@@ -169,11 +169,11 @@ function Section(props: {
} }
function Sec3Item(props: { function Sec3Item(props: {
icon: string, icon: string
title: string, title: string
terms: { terms: {
icon: string, icon: string
text: string, text: string
}[] }[]
}) { }) {
return ( return (