Compare commits

...

8 Commits

Author SHA1 Message Date
b10dbcb23f fix(ci): Persist Next.js BuildKit cache mount to runner stage
Some checks failed
Build & Deploy / 🔍 Prepare (push) Successful in 15s
Build & Deploy / 🧪 QA (push) Successful in 9m12s
Build & Deploy / 🚀 Deploy (push) Has been cancelled
Build & Deploy / 🧪 Smoke Test (push) Has been cancelled
Build & Deploy / ⚡ Lighthouse (push) Has been cancelled
Build & Deploy / ♿ WCAG (push) Has been cancelled
Build & Deploy / 🛡️ Quality Gates (push) Has been cancelled
Build & Deploy / 🔔 Notify (push) Has been cancelled
Build & Deploy / 🏗️ Build (push) Has started running
2026-02-23 02:26:41 +01:00
65bb9c620a chore(ci): Fix TURBO_TELEMETRY_DISABLED variable type
Some checks failed
Build & Deploy / 🔍 Prepare (push) Has started running
Build & Deploy / 🧪 QA (push) Has been cancelled
Build & Deploy / 🏗️ Build (push) Has been cancelled
Build & Deploy / 🚀 Deploy (push) Has been cancelled
Build & Deploy / 🧪 Smoke Test (push) Has been cancelled
Build & Deploy / ⚡ Lighthouse (push) Has been cancelled
Build & Deploy / ♿ WCAG (push) Has been cancelled
Build & Deploy / 🛡️ Quality Gates (push) Has been cancelled
Build & Deploy / 🔔 Notify (push) Has been cancelled
2026-02-23 02:25:49 +01:00
63853ffa89 chore(ci): Optimize Turborepo cache restoring and add Next.js BuildKit caching
Some checks failed
Build & Deploy / 🔍 Prepare (push) Successful in 29s
Build & Deploy / 🚀 Deploy (push) Has been cancelled
Build & Deploy / 🧪 Smoke Test (push) Has been cancelled
Build & Deploy / ⚡ Lighthouse (push) Has been cancelled
Build & Deploy / ♿ WCAG (push) Has been cancelled
Build & Deploy / 🛡️ Quality Gates (push) Has been cancelled
Build & Deploy / 🔔 Notify (push) Has been cancelled
Build & Deploy / 🧪 QA (push) Successful in 1m45s
Build & Deploy / 🏗️ Build (push) Has been cancelled
2026-02-23 02:17:31 +01:00
9694c77ef7 fix(infra): add explicit staging url mapping for local imgproxy bypass
Some checks failed
Build & Deploy / 🔍 Prepare (push) Successful in 46s
Build & Deploy / 🚀 Deploy (push) Has been cancelled
Build & Deploy / 🧪 Smoke Test (push) Has been cancelled
Build & Deploy / ⚡ Lighthouse (push) Has been cancelled
Build & Deploy / ♿ WCAG (push) Has been cancelled
Build & Deploy / 🛡️ Quality Gates (push) Has been cancelled
Build & Deploy / 🔔 Notify (push) Has been cancelled
Build & Deploy / 🧪 QA (push) Successful in 1m51s
Build & Deploy / 🏗️ Build (push) Has been cancelled
2026-02-23 02:09:28 +01:00
2c11b5026a chore(ci): Use Turborepo for CI pipeline QA caching
Some checks failed
Build & Deploy / 🔍 Prepare (push) Successful in 26s
Build & Deploy / 🏗️ Build (push) Has been cancelled
Build & Deploy / 🚀 Deploy (push) Has been cancelled
Build & Deploy / 🧪 Smoke Test (push) Has been cancelled
Build & Deploy / ⚡ Lighthouse (push) Has been cancelled
Build & Deploy / ♿ WCAG (push) Has been cancelled
Build & Deploy / 🧪 QA (push) Has been cancelled
Build & Deploy / 🛡️ Quality Gates (push) Has been cancelled
Build & Deploy / 🔔 Notify (push) Has been cancelled
2026-02-23 02:00:48 +01:00
eaa90c65f1 chore: ignore public/uploads and remove images from tracking
Some checks failed
Build & Deploy / 🔍 Prepare (push) Successful in 19s
Build & Deploy / 🧪 QA (push) Successful in 2m33s
Build & Deploy / 🏗️ Build (push) Successful in 4m55s
Build & Deploy / 🚀 Deploy (push) Successful in 1m29s
Build & Deploy / 🧪 Smoke Test (push) Successful in 1m22s
Build & Deploy / ⚡ Lighthouse (push) Successful in 2m42s
Build & Deploy / 🛡️ Quality Gates (push) Successful in 3m14s
Build & Deploy / 🔔 Notify (push) Has been cancelled
Build & Deploy / ♿ WCAG (push) Has been cancelled
2026-02-23 01:42:37 +01:00
2a47d22e26 feat: change img proxy configuration
All checks were successful
Build & Deploy / 🔍 Prepare (push) Successful in 8s
Build & Deploy / 🧪 QA (push) Successful in 1m56s
Build & Deploy / 🏗️ Build (push) Successful in 4m5s
Build & Deploy / 🚀 Deploy (push) Successful in 1m19s
Build & Deploy / 🧪 Smoke Test (push) Successful in 48s
Build & Deploy / 🛡️ Quality Gates (push) Successful in 1m39s
Build & Deploy / ⚡ Lighthouse (push) Successful in 5m51s
Build & Deploy / ♿ WCAG (push) Successful in 7m0s
Build & Deploy / 🔔 Notify (push) Successful in 2s
2026-02-23 01:40:13 +01:00
33d2d67774 feat: Improve error handling, refine i18n title fallbacks for product pages, update Next.js type paths, and add an image loader test file.
All checks were successful
Build & Deploy / 🔍 Prepare (push) Successful in 12s
Build & Deploy / 🧪 QA (push) Successful in 1m58s
Build & Deploy / 🏗️ Build (push) Successful in 5m44s
Build & Deploy / 🚀 Deploy (push) Successful in 27s
Build & Deploy / 🧪 Smoke Test (push) Successful in 48s
Build & Deploy / 🛡️ Quality Gates (push) Successful in 1m39s
Build & Deploy / ⚡ Lighthouse (push) Successful in 5m39s
Build & Deploy / ♿ WCAG (push) Successful in 5m35s
Build & Deploy / 🔔 Notify (push) Successful in 2s
2026-02-22 18:30:44 +01:00
17 changed files with 186 additions and 60 deletions

View File

@@ -1,5 +1,15 @@
node_modules node_modules
.next .next
.DS_Store
.git
.gitignore
.gitea
.github
public/uploads
directus/uploads
.turbo
reference/
.next
!.next/cache !.next/cache
.git .git
.DS_Store .DS_Store

3
.env
View File

@@ -35,4 +35,5 @@ GATEKEEPER_PASSWORD=klz2026
COOKIE_DOMAIN=localhost COOKIE_DOMAIN=localhost
INFRA_DIRECTUS_URL=http://localhost:8059 INFRA_DIRECTUS_URL=http://localhost:8059
INFRA_DIRECTUS_TOKEN=59fb8f4c1a51b18fe28ad947f713914e INFRA_DIRECTUS_TOKEN=59fb8f4c1a51b18fe28ad947f713914e
GATEKEEPER_ORIGIN=http://klz.localhost GATEKEEPER_ORIGIN=http://klz.localhost
OPENROUTER_API_KEY=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

View File

@@ -46,8 +46,19 @@ jobs:
env: env:
NPM_TOKEN: ${{ secrets.REGISTRY_PASS }} NPM_TOKEN: ${{ secrets.REGISTRY_PASS }}
- name: Setup Turbo cache
uses: actions/cache@v4
with:
path: .turbo
key: ${{ runner.os }}-turbo-${{ github.ref_name }}-${{ github.sha }}
restore-keys: |
${{ runner.os }}-turbo-${{ github.ref_name }}-
${{ runner.os }}-turbo-
- name: 🧪 QA Checks - name: 🧪 QA Checks
run: pnpm check:mdx && pnpm lint && pnpm typecheck && pnpm test env:
TURBO_TELEMETRY_DISABLED: "1"
run: npx turbo run check:mdx lint typecheck test --cache-dir=".turbo"
- name: 🏗️ Build - name: 🏗️ Build
run: pnpm build run: pnpm build

View File

@@ -171,15 +171,22 @@ jobs:
echo "//${{ vars.REGISTRY_HOST || 'npm.infra.mintel.me' }}/:_authToken=${{ secrets.REGISTRY_PASS }}" >> .npmrc echo "//${{ vars.REGISTRY_HOST || 'npm.infra.mintel.me' }}/:_authToken=${{ secrets.REGISTRY_PASS }}" >> .npmrc
- name: Install dependencies - name: Install dependencies
run: pnpm install --frozen-lockfile run: pnpm install --frozen-lockfile
- name: Setup Turbo cache
uses: actions/cache@v4
with:
path: .turbo
key: ${{ runner.os }}-turbo-${{ github.ref_name }}-${{ github.sha }}
restore-keys: |
${{ runner.os }}-turbo-${{ github.ref_name }}-
${{ runner.os }}-turbo-
- name: 🔒 Security Audit - name: 🔒 Security Audit
run: pnpm audit --audit-level high run: pnpm audit --audit-level high
- name: 🧪 QA Checks - name: 🧪 QA Checks
if: github.event.inputs.skip_checks != 'true' if: github.event.inputs.skip_checks != 'true'
run: | env:
pnpm lint TURBO_TELEMETRY_DISABLED: "1"
pnpm check:spell run: npx turbo run lint check:spell typecheck test --cache-dir=".turbo"
pnpm typecheck
pnpm test
# ────────────────────────────────────────────────────────────────────────────── # ──────────────────────────────────────────────────────────────────────────────
# JOB 3: Build & Push # JOB 3: Build & Push
@@ -205,6 +212,8 @@ jobs:
push: true push: true
provenance: false provenance: false
platforms: linux/arm64 platforms: linux/arm64
cache-from: type=gha,scope=nextjs-build-${{ needs.prepare.outputs.target }}
cache-to: type=gha,mode=max,scope=nextjs-build-${{ needs.prepare.outputs.target }}
build-args: | build-args: |
NEXT_PUBLIC_BASE_URL=${{ needs.prepare.outputs.next_public_url }} NEXT_PUBLIC_BASE_URL=${{ needs.prepare.outputs.next_public_url }}
NEXT_PUBLIC_TARGET=${{ needs.prepare.outputs.target }} NEXT_PUBLIC_TARGET=${{ needs.prepare.outputs.target }}

10
.gitignore vendored
View File

@@ -1,6 +1,7 @@
node_modules node_modules
.next .next
.DS_Store .DS_Store
public/uploads
# Lighthouse CI # Lighthouse CI
.lighthouseci/ .lighthouseci/
@@ -18,4 +19,11 @@ directus/uploads
# Pa11y CI # Pa11y CI
.pa11yci/ .pa11yci/
.htmlvalidate-tmp .htmlvalidate-tmp
# Turborepo
.turbo
# Test Outputs
html-errors*.json
reference/

View File

@@ -41,7 +41,10 @@ CMD ["pnpm", "dev:local"]
# Build application # Build application
# Stage 3: Builder (Production) # Stage 3: Builder (Production)
FROM base AS builder FROM base AS builder
RUN pnpm build RUN --mount=type=cache,target=/app/.next/cache,id=nextjs-cache \
pnpm build && \
mkdir -p /app/.next-cache-tmp && \
cp -r /app/.next/cache/* /app/.next-cache-tmp/ || true
# Stage 3: Runner # Stage 3: Runner
FROM registry.infra.mintel.me/mintel/runtime:v1.7.10 AS runner FROM registry.infra.mintel.me/mintel/runtime:v1.7.10 AS runner
@@ -60,6 +63,6 @@ ENV NODE_ENV=production
COPY --from=builder --chown=nextjs:nodejs /app/public ./public COPY --from=builder --chown=nextjs:nodejs /app/public ./public
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
COPY --from=builder --chown=nextjs:nodejs /app/.next/cache ./.next/cache COPY --from=builder --chown=nextjs:nodejs /app/.next-cache-tmp ./.next/cache
CMD ["node", "server.js"] CMD ["node", "server.js"]

View File

@@ -16,12 +16,18 @@ export default function Error({
const t = useTranslations('Error'); const t = useTranslations('Error');
useEffect(() => { useEffect(() => {
// Treat "Failed to find Server Action" as a deployment sync issue and reload
if (error?.message?.includes('Failed to find Server Action')) {
window.location.reload();
return;
}
const services = getAppServices(); const services = getAppServices();
services.errors.captureException(error); services.errors.captureException(error);
services.logger.error('Application error caught by boundary', { services.logger.error('Application error caught by boundary', {
message: error.message, message: error?.message || 'Unknown error',
stack: error.stack, stack: error?.stack,
digest: error.digest digest: error?.digest,
}); });
}, [error]); }, [error]);
@@ -36,19 +42,14 @@ export default function Error({
<Heading level={1} className="text-6xl md:text-8xl font-bold mb-2 text-saturated"> <Heading level={1} className="text-6xl md:text-8xl font-bold mb-2 text-saturated">
500 500
</Heading> </Heading>
<Scribble <Scribble variant="underline" className="w-full h-6 -bottom-2 left-0 text-saturated/40" />
variant="underline"
className="w-full h-6 -bottom-2 left-0 text-saturated/40"
/>
</div> </div>
<Heading level={2} className="text-2xl md:text-3xl font-bold mb-4"> <Heading level={2} className="text-2xl md:text-3xl font-bold mb-4">
{t('title')} {t('title')}
</Heading> </Heading>
<p className="text-white/60 mb-10 max-w-md text-lg"> <p className="text-white/60 mb-10 max-w-md text-lg">{t('description')}</p>
{t('description')}
</p>
<div className="flex flex-col sm:flex-row gap-4"> <div className="flex flex-col sm:flex-row gap-4">
<Button onClick={() => reset()} variant="saturated" size="lg"> <Button onClick={() => reset()} variant="saturated" size="lg">

View File

@@ -223,7 +223,7 @@ export default async function ProductPage({ params }: ProductPageProps) {
href={`/${locale}/${await mapFileSlugToTranslated('products', locale)}`} href={`/${locale}/${await mapFileSlugToTranslated('products', locale)}`}
className="hover:text-accent transition-colors" className="hover:text-accent transition-colors"
> >
{t.has('breadcrumb') ? t('breadcrumb') : t('title').replace(/<[^>]*>/g, '')} {t.has('breadcrumb') ? t('breadcrumb') : 'Products'}
</Link> </Link>
<span className="mx-3 opacity-30">/</span> <span className="mx-3 opacity-30">/</span>
<span className="text-white/90">{categoryTitle}</span> <span className="text-white/90">{categoryTitle}</span>
@@ -384,7 +384,7 @@ export default async function ProductPage({ params }: ProductPageProps) {
href={`/${locale}/${await mapFileSlugToTranslated('products', locale)}`} href={`/${locale}/${await mapFileSlugToTranslated('products', locale)}`}
className="hover:text-accent transition-colors" className="hover:text-accent transition-colors"
> >
{t.has('breadcrumb') ? t('breadcrumb') : t('title').replace(/<[^>]*>/g, '')} {t.has('breadcrumb') ? t('breadcrumb') : 'Products'}
</Link> </Link>
<span className="mx-4 opacity-20">/</span> <span className="mx-4 opacity-20">/</span>
<Link <Link

View File

@@ -12,7 +12,11 @@ export default async function Image({ params }: { params: Promise<{ locale: stri
const t = await getTranslations({ locale, namespace: 'Products' }); const t = await getTranslations({ locale, namespace: 'Products' });
const fonts = await getOgFonts(); const fonts = await getOgFonts();
const title = t('meta.title') || t('breadcrumb') || t('title').replace(/<[^>]*>/g, ''); const title = t.has('meta.title')
? t('meta.title')
: t.has('breadcrumb')
? t('breadcrumb')
: 'Products';
const description = t('meta.description') || t('subtitle'); const description = t('meta.description') || t('subtitle');
return new ImageResponse( return new ImageResponse(

View File

@@ -17,7 +17,11 @@ interface ProductsPageProps {
export async function generateMetadata({ params }: ProductsPageProps): Promise<Metadata> { export async function generateMetadata({ params }: ProductsPageProps): Promise<Metadata> {
const { locale } = await params; const { locale } = await params;
const t = await getTranslations({ locale, namespace: 'Products' }); const t = await getTranslations({ locale, namespace: 'Products' });
const title = t('meta.title') || t('breadcrumb') || t('title').replace(/<[^>]*>/g, ''); const title = t.has('meta.title')
? t('meta.title')
: t.has('breadcrumb')
? t('breadcrumb')
: 'Products';
const description = t('meta.description') || t('subtitle'); const description = t('meta.description') || t('subtitle');
return { return {
title, title,

View File

@@ -43,7 +43,11 @@ export default function ProductCategories() {
return ( return (
<Section className="bg-neutral-light py-0 md:py-0 lg:py-0 -mt-px"> <Section className="bg-neutral-light py-0 md:py-0 lg:py-0 -mt-px">
{t('title') && <h2 className="sr-only">{t('title')}</h2>} {t.has('title') && (
<h2 className="sr-only">
{t.rich('title', { green: (chunks: any) => <span>{chunks}</span> })}
</h2>
)}
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4"> <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4">
{categories.map((category, idx) => ( {categories.map((category, idx) => (
<Link <Link

View File

@@ -155,7 +155,7 @@ services:
- default - default
klz-imgproxy: klz-imgproxy:
image: darthsim/imgproxy:latest image: registry.infra.mintel.me/mintel/image-processor:latest
restart: unless-stopped restart: unless-stopped
networks: networks:
- default - default
@@ -165,12 +165,10 @@ services:
- "cms.klz.localhost:host-gateway" - "cms.klz.localhost:host-gateway"
- "host.docker.internal:host-gateway" - "host.docker.internal:host-gateway"
environment: environment:
IMGPROXY_URL_MAPPING: "${NEXT_PUBLIC_BASE_URL}:http://klz-app:3000,${DIRECTUS_URL}:http://klz-cms:8055" OPENROUTER_API_KEY: ${OPENROUTER_API_KEY}
IMGPROXY_USE_ETAG: "true" # explicitly map localhost, production and staging to bypass gatekeeper
IMGPROXY_MAX_SRC_RESOLUTION: 20 IMGPROXY_URL_MAPPING: "https://staging.klz-cables.com:http://klz-app:3000,https://klz-cables.com:http://klz-app:3000,https://${TRAEFIK_HOST:-klz.localhost}:http://klz-app:3000,${DIRECTUS_URL}:http://klz-cms:8055,https://cms.klz-cables.com:http://klz-cms:8055"
IMGPROXY_IGNORE_SSL_ERRORS: "true"
IMGPROXY_LOG_LEVEL: debug IMGPROXY_LOG_LEVEL: debug
IMGPROXY_ALLOW_LOCAL_NETWORKS: "true"
labels: labels:
- "traefik.enable=true" - "traefik.enable=true"

View File

@@ -39,34 +39,21 @@ export function getImgproxyUrl(src: string, options: ImgproxyOptions = {}): stri
// Also handle direct container names if needed // Also handle direct container names if needed
} }
const { const { width = 0, height = 0, enlarge = false, extension = '' } = options;
width = 0,
height = 0,
resizing_type = 'fit',
gravity = 'sm', // Default to smart gravity
enlarge = false,
extension = '',
} = options;
// Processing options let quality = 80;
// Format: /rs:<type>:<width>:<height>:<enlarge>/g:<gravity> if (extension) quality = 90;
const processingOptions = [
`rs:${resizing_type}:${width}:${height}:${enlarge ? 1 : 0}`,
`g:${gravity}`,
].join('/');
// Using Base64 encoding for the source URL. // Re-map imgproxy URL to our new parameter structure
// This completely eliminates any risk of intermediate proxies (Traefik/Next.js) // e.g. /process?url=...&w=...&h=...&q=...&format=...
// URL-decoding the path, which corrupts the double-slash (// to /) and causes 403 errors. const queryParams = new URLSearchParams({
// Imgproxy expects URL-safe Base64 (RFC 4648) without padding. url: absoluteSrc,
const b64 = });
typeof window === 'undefined'
? Buffer.from(absoluteSrc).toString('base64')
: btoa(unescape(encodeURIComponent(absoluteSrc)));
const urlSafeB64 = b64.replace(/\+/g, '-').replace(/\//g, '_').replace(/=/g, ''); if (width > 0) queryParams.set('w', width.toString());
if (height > 0) queryParams.set('h', height.toString());
if (extension) queryParams.set('format', extension.replace('.', ''));
if (quality) queryParams.set('q', quality.toString());
const suffix = extension ? `.${extension}` : ''; return `${baseUrl}/process?${queryParams.toString()}`;
return `${baseUrl}/unsafe/${processingOptions}/${urlSafeB64}${suffix}`;
} }

2
next-env.d.ts vendored
View File

@@ -1,6 +1,6 @@
/// <reference types="next" /> /// <reference types="next" />
/// <reference types="next/image-types/global" /> /// <reference types="next/image-types/global" />
import "./.next/dev/types/routes.d.ts"; import "./.next/types/routes.d.ts";
// NOTE: This file should not be edited // NOTE: This file should not be edited
// see https://nextjs.org/docs/app/api-reference/config/typescript for more information. // see https://nextjs.org/docs/app/api-reference/config/typescript for more information.

View File

@@ -2,6 +2,7 @@
"name": "klz-cables-nextjs", "name": "klz-cables-nextjs",
"type": "module", "type": "module",
"private": true, "private": true,
"packageManager": "pnpm@10.18.3",
"dependencies": { "dependencies": {
"@directus/sdk": "^21.0.0", "@directus/sdk": "^21.0.0",
"@mintel/mail": "1.8.3", "@mintel/mail": "1.8.3",
@@ -79,6 +80,7 @@
"start-server-and-test": "^2.1.3", "start-server-and-test": "^2.1.3",
"tailwindcss": "^4.1.18", "tailwindcss": "^4.1.18",
"tsx": "^4.21.0", "tsx": "^4.21.0",
"turbo": "^2.8.10",
"typescript": "^5.7.2", "typescript": "^5.7.2",
"vitest": "^4.0.16" "vitest": "^4.0.16"
}, },
@@ -124,7 +126,7 @@
"prepare": "husky", "prepare": "husky",
"preinstall": "npx only-allow pnpm" "preinstall": "npx only-allow pnpm"
}, },
"version": "1.0.0", "version": "1.0.1-rc.0",
"pnpm": { "pnpm": {
"overrides": { "overrides": {
"next": "16.1.6", "next": "16.1.6",

64
pnpm-lock.yaml generated
View File

@@ -235,6 +235,9 @@ importers:
tsx: tsx:
specifier: ^4.21.0 specifier: ^4.21.0
version: 4.21.0 version: 4.21.0
turbo:
specifier: ^2.8.10
version: 2.8.10
typescript: typescript:
specifier: ^5.7.2 specifier: ^5.7.2
version: 5.9.3 version: 5.9.3
@@ -7166,6 +7169,40 @@ packages:
engines: {node: '>=18.0.0'} engines: {node: '>=18.0.0'}
hasBin: true hasBin: true
turbo-darwin-64@2.8.10:
resolution: {integrity: sha512-A03fXh+B7S8mL3PbdhTd+0UsaGrhfyPkODvzBDpKRY7bbeac4MDFpJ7I+Slf2oSkCEeSvHKR7Z4U71uKRUfX7g==}
cpu: [x64]
os: [darwin]
turbo-darwin-arm64@2.8.10:
resolution: {integrity: sha512-sidzowgWL3s5xCHLeqwC9M3s9M0i16W1nuQF3Mc7fPHpZ+YPohvcbVFBB2uoRRHYZg6yBnwD4gyUHKTeXfwtXA==}
cpu: [arm64]
os: [darwin]
turbo-linux-64@2.8.10:
resolution: {integrity: sha512-YK9vcpL3TVtqonB021XwgaQhY9hJJbKKUhLv16osxV0HkcQASQWUqR56yMge7puh6nxU67rQlTq1b7ksR1T3KA==}
cpu: [x64]
os: [linux]
turbo-linux-arm64@2.8.10:
resolution: {integrity: sha512-3+j2tL0sG95iBJTm+6J8/45JsETQABPqtFyYjVjBbi6eVGdtNTiBmHNKrbvXRlQ3ZbUG75bKLaSSDHSEEN+btQ==}
cpu: [arm64]
os: [linux]
turbo-windows-64@2.8.10:
resolution: {integrity: sha512-hdeF5qmVY/NFgiucf8FW0CWJWtyT2QPm5mIsX0W1DXAVzqKVXGq+Zf+dg4EUngAFKjDzoBeN6ec2Fhajwfztkw==}
cpu: [x64]
os: [win32]
turbo-windows-arm64@2.8.10:
resolution: {integrity: sha512-QGdr/Q8LWmj+ITMkSvfiz2glf0d7JG0oXVzGL3jxkGqiBI1zXFj20oqVY0qWi+112LO9SVrYdpHS0E/oGFrMbQ==}
cpu: [arm64]
os: [win32]
turbo@2.8.10:
resolution: {integrity: sha512-OxbzDES66+x7nnKGg2MwBA1ypVsZoDTLHpeaP4giyiHSixbsiTaMyeJqbEyvBdp5Cm28fc+8GG6RdQtic0ijwQ==}
hasBin: true
type-check@0.4.0: type-check@0.4.0:
resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==} resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==}
engines: {node: '>= 0.8.0'} engines: {node: '>= 0.8.0'}
@@ -15498,6 +15535,33 @@ snapshots:
optionalDependencies: optionalDependencies:
fsevents: 2.3.3 fsevents: 2.3.3
turbo-darwin-64@2.8.10:
optional: true
turbo-darwin-arm64@2.8.10:
optional: true
turbo-linux-64@2.8.10:
optional: true
turbo-linux-arm64@2.8.10:
optional: true
turbo-windows-64@2.8.10:
optional: true
turbo-windows-arm64@2.8.10:
optional: true
turbo@2.8.10:
optionalDependencies:
turbo-darwin-64: 2.8.10
turbo-darwin-arm64: 2.8.10
turbo-linux-64: 2.8.10
turbo-linux-arm64: 2.8.10
turbo-windows-64: 2.8.10
turbo-windows-arm64: 2.8.10
type-check@0.4.0: type-check@0.4.0:
dependencies: dependencies:
prelude-ls: 1.2.1 prelude-ls: 1.2.1

20
turbo.json Normal file
View File

@@ -0,0 +1,20 @@
{
"$schema": "https://turbo.build/schema.json",
"tasks": {
"lint": {
"outputs": []
},
"typecheck": {
"outputs": []
},
"test": {
"outputs": []
},
"check:spell": {
"outputs": []
},
"check:mdx": {
"outputs": []
}
}
}