diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml
new file mode 100644
index 0000000..4ef8272
--- /dev/null
+++ b/.github/workflows/build-and-test.yml
@@ -0,0 +1,41 @@
+name: Build and Test Next.js site
+
+on:
+ push:
+ branches: [ main ]
+ pull_request:
+ branches: [ '*' ] # Wildcard matches all branches
+
+ workflow_dispatch:
+
+jobs:
+ build-and-test:
+ runs-on: ubuntu-latest
+
+ steps:
+ - uses: actions/checkout@v3
+
+ - name: Setup Node.js
+ uses: actions/setup-node@v3
+ with:
+ node-version: '23'
+ cache: 'npm'
+
+ - name: Install dependencies
+ run: npm ci
+
+ - name: Build
+ run: npm run build
+
+ - name: Test
+ run: npm run test
+
+ - name: Cache Next.js build
+ uses: actions/cache@v3
+ with:
+ path: |
+ ~/.npm
+ .next/cache
+ key: ${{ runner.os }}-nextjs-${{ hashFiles('**/package-lock.json') }}
+ restore-keys: |
+ ${{ runner.os }}-nextjs-
\ No newline at end of file
diff --git a/.github/workflows/docker-image.yml b/.github/workflows/docker-image.yml
deleted file mode 100644
index 374d20c..0000000
--- a/.github/workflows/docker-image.yml
+++ /dev/null
@@ -1,63 +0,0 @@
-
-name: Create and publish a Docker image
-
-# Configures this workflow to run every time a change is pushed to the branch called `release`.
-on:
- push:
- branches: ['main']
-
-# Defines two custom environment variables for the workflow. These are used for the Container registry domain, and a name for the Docker image that this workflow builds.
-env:
- REGISTRY: ghcr.io
- IMAGE_NAME: ${{ github.repository }}
-
-# There is a single job in this workflow. It's configured to run on the latest available version of Ubuntu.
-jobs:
- build-and-push-image:
- runs-on: ubuntu-latest
- permissions:
- contents: read
- packages: write
- attestations: write
- id-token: write
-
- steps:
- - name: Checkout repository
- uses: actions/checkout@v4
-
- - name: Log in to the Container registry
- uses: docker/login-action@65b78e6e13532edd9afa3aa52ac7964289d1a9c1
- with:
- registry: ${{ env.REGISTRY }}
- username: ${{ github.actor }}
- password: ${{ secrets.GITHUB_TOKEN }}
-
- - name: Extract metadata (tags, labels) for Docker
- id: meta
- uses: docker/metadata-action@9ec57ed1fcdbf14dcef7dfbe97b2010124a938b7
- with:
- images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
-
- - name: Set up QEMU
- uses: docker/setup-qemu-action@v3
-
- - name: Set up Docker Buildx
- uses: docker/setup-buildx-action@v3
-
- - name: Build and push Docker image
- id: push
- uses: docker/build-push-action@v6
- with:
- context: .
- platforms: linux/amd64,linux/arm64
- push: true
- tags: ${{ steps.meta.outputs.tags }}
- labels: ${{ steps.meta.outputs.labels }}
-
- - name: Generate artifact attestation
- uses: actions/attest-build-provenance@v1
- with:
- subject-name: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME}}
- subject-digest: ${{ steps.push.outputs.digest }}
- push-to-registry: true
-
diff --git a/__test__/components/Article.test.jsx b/__test__/components/Article.test.jsx
deleted file mode 100644
index a8ba778..0000000
--- a/__test__/components/Article.test.jsx
+++ /dev/null
@@ -1,46 +0,0 @@
-/* eslint-env jest */
-import { render } from '@testing-library/react'
-import Article from '../../src/components/Article/Article'
-import '@testing-library/jest-dom'
-import { formatDate } from '@/lib/helpers'
-
-describe('Article', () => {
- it('renders title', () => {
- const props = generateArticleProps()
- const { getByText } = render()
- const titleElement = getByText(props.attributes.title)
- expect(titleElement).toBeInTheDocument()
- })
-
- it('renders description', () => {
- const props = generateArticleProps()
- const { getByText } = render()
- const descriptionElement = getByText(props.attributes.desc)
- expect(descriptionElement).toBeInTheDocument()
- })
-
- it('renders pubdate if available', () => {
- const props = generateArticleProps()
- const { getByText } = render()
- const pubdateElement = getByText(formatDate(props.attributes.pubdate))
- expect(pubdateElement).toBeInTheDocument()
- })
-
- it('renders content', () => {
- const props = generateArticleProps()
- const { container } = render()
- const contentElement = container.querySelector('[data-test=content]')
- expect(contentElement.innerHTML).toBe(props.html)
- })
-})
-
-function generateArticleProps () {
- return {
- attributes: {
- title: 'My title',
- desc: 'My description',
- pubdate: new Date().toUTCString()
- },
- html: '
This is my content!
'
- }
-}
diff --git a/__test__/components/ExsternalLink.test.jsx b/__test__/components/ExsternalLink.test.jsx
deleted file mode 100644
index 506d457..0000000
--- a/__test__/components/ExsternalLink.test.jsx
+++ /dev/null
@@ -1,28 +0,0 @@
-/* eslint-env jest */
-import { render, screen } from '@testing-library/react'
-import ExternalLink from '../../src/components/ExternalLink/ExternalLink'
-import '@testing-library/jest-dom'
-
-describe('ExternalLink', () => {
- const props = {
- href: 'https://example.com',
- children: 'Test Link'
- }
-
- it('renders without crashing', () => {
- render()
- })
-
- it('renders correct href and rel attributes', () => {
- render()
- const link = screen.getByText(props.children)
- expect(link).toHaveAttribute('href', props.href)
- expect(link).toHaveAttribute('rel', 'nofollow noopener')
- expect(link).toHaveAttribute('target', '_blank')
- })
-
- it('renders children correctly', () => {
- render()
- expect(screen.getByText(props.children)).toBeInTheDocument()
- })
-})
diff --git a/__test__/components/Footer.test.jsx b/__test__/components/Footer.test.jsx
deleted file mode 100644
index b6882e6..0000000
--- a/__test__/components/Footer.test.jsx
+++ /dev/null
@@ -1,10 +0,0 @@
-/* eslint-env jest */
-import { render } from '@testing-library/react'
-import '@testing-library/jest-dom'
-import Footer from '@/components/Footer/Footer'
-
-describe('Footer', () => {
- it('renders without crashing', () => {
- render()
- })
-})
diff --git a/__test__/components/Grid.test.jsx b/__test__/components/Grid.test.jsx
deleted file mode 100644
index 4594a17..0000000
--- a/__test__/components/Grid.test.jsx
+++ /dev/null
@@ -1,16 +0,0 @@
-/* eslint-env jest */
-import { render } from '@testing-library/react'
-import '@testing-library/jest-dom'
-import Grid from '@/components/Grid/Grid'
-
-describe('Grid', () => {
- it('renders without crashing', () => {
- const { container } = render()
- expect(container.firstChild).toHaveClass('grid')
- })
-
- it('renders its children', () => {
- const { getByText } = render(Child
)
- expect(getByText('Child')).toBeInTheDocument()
- })
-})
diff --git a/__test__/components/Header.test.jsx b/__test__/components/Header.test.jsx
deleted file mode 100644
index 200a48a..0000000
--- a/__test__/components/Header.test.jsx
+++ /dev/null
@@ -1,19 +0,0 @@
-/* eslint-env jest */
-import { render, screen } from '@testing-library/react'
-import Header from '../../src/components/Header/Header'
-import '@testing-library/jest-dom'
-
-describe('Header', () => {
- it('renders without crashing', () => {
- render(
- )
- })
-
- it('renders correct navigation links', () => {
- render()
- const links = ['Home', 'Writing', 'CV']
- links.forEach(link => {
- expect(screen.getByText(link)).toBeInTheDocument()
- })
- })
-})
diff --git a/__test__/components/Resume.test.jsx b/__test__/components/Resume.test.jsx
deleted file mode 100644
index 5a56848..0000000
--- a/__test__/components/Resume.test.jsx
+++ /dev/null
@@ -1,63 +0,0 @@
-/* eslint-env jest */
-import { render } from '@testing-library/react'
-import Resume from '../../src/components/Resume/Resume'
-import '@testing-library/jest-dom'
-
-describe('Resume', () => {
- const props = {
- competencies: ['Competency 1', 'Competency 2'],
- education: 'My education history',
- certifications: ['Certification 1', 'Certification 2'],
- languages: [{ name: 'English', proficiency: 'Fluent' }],
- experience: [
- {
- employer: 'Employer 1',
- position: 'Position 1',
- start: 'Start date 1',
- end: 'End date 1',
- desc: 'Description 1'
- }
- ]
- }
-
- it('renders without crashing', () => {
- render()
- })
-
- it('renders competencies', () => {
- const { getByText } = render()
- props.competencies.forEach(competency => {
- expect(getByText(competency)).toBeInTheDocument()
- })
- })
-
- it('renders education', () => {
- const { getByText } = render()
- expect(getByText(props.education)).toBeInTheDocument()
- })
-
- it('renders certifications', () => {
- const { getByText } = render()
- props.certifications.forEach(certification => {
- expect(getByText(certification)).toBeInTheDocument()
- })
- })
-
- it('renders languages', () => {
- const { getByText } = render()
- props.languages.forEach(language => {
- expect(getByText(`${language.name} - ${language.proficiency}`)).toBeInTheDocument()
- })
- })
-
- it('renders experience', () => {
- const { getByText } = render()
- props.experience.forEach(exp => {
- expect(getByText(exp.employer)).toBeInTheDocument()
- expect(getByText(exp.position)).toBeInTheDocument()
- expect(getByText(exp.start)).toBeInTheDocument()
- expect(getByText(exp.end)).toBeInTheDocument()
- expect(getByText(exp.desc)).toBeInTheDocument()
- })
- })
-})
diff --git a/__test__/layouts/DefaultLayout.test.jsx b/__test__/layouts/DefaultLayout.test.jsx
deleted file mode 100644
index 5cecb03..0000000
--- a/__test__/layouts/DefaultLayout.test.jsx
+++ /dev/null
@@ -1,29 +0,0 @@
-/* eslint-env jest */
-import { render, screen } from '@testing-library/react'
-import DefaultLayout from '../../src/layouts/DefaultLayout/DefaultLayout'
-import '@testing-library/jest-dom'
-
-describe('DefaultLayout', () => {
- const props = {
- children: Test Content
- }
-
- it('renders without crashing', () => {
- render()
- })
-
- it('renders Header', () => {
- render()
- expect(screen.getByTestId('header')).toBeInTheDocument()
- })
-
- it('renders Footer', () => {
- render()
- expect(screen.getByTestId('footer')).toBeInTheDocument()
- })
-
- it('renders children correctly', () => {
- render()
- expect(screen.getByText('Test Content')).toBeInTheDocument()
- })
-})
diff --git a/__test__/pages/Writing.test.jsx b/__test__/pages/Writing.test.jsx
deleted file mode 100644
index 126f220..0000000
--- a/__test__/pages/Writing.test.jsx
+++ /dev/null
@@ -1,69 +0,0 @@
-/* eslint-env jest */
-import { render } from '@testing-library/react'
-import '@testing-library/jest-dom'
-import Writing, { Title, Description } from '../../src/pages/writing'
-
-jest.mock('next/head', () => {
- return {
- __esModule: true,
- default: ({ children }) => {
- return <>{children}>
- }
- }
-})
-
-describe('Writing', () => {
- it('renders without crashing', () => {
- render(
- getStubWritingComponent())
- })
-
- it('renders meta title', async () => {
- const { container } = render(getStubWritingComponent())
- expect(container.querySelector('title')).toHaveTextContent(Title)
- })
-
- it('renders opengraph title', () => {
- const { container } = render(getStubWritingComponent())
- expect(container.querySelector('meta[property="og:title"]'))
- .toHaveAttribute('content', Title)
- })
-
- it('renders meta description', () => {
- const { container } = render(getStubWritingComponent())
- expect(container.querySelector('meta[name="description"]'))
- .toHaveAttribute('content', Description)
- })
-
- it('renders opengraph description', () => {
- const { container } = render(getStubWritingComponent())
-
- expect(container.querySelector('meta[property="og:description"]'))
- .toHaveAttribute('content', Description)
- })
-})
-
-function getStubWritingComponent () {
- const entries = [
- {
- attributes: {
- title: 'My title one',
- pubdate: 'Mon, 18 Mar 2024 16:47:32 GMT',
- desc: 'This is my description.'
- },
- html: 'This is some content
',
- slug: 'my-title-one'
- },
- {
- attributes: {
- title: 'My title Two',
- pubdate: 'Mon, 19 Mar 2024 16:47:32 GMT',
- desc: 'This is my description.'
- },
- html: 'This is some content
',
- slug: 'my-title-two'
- }
- ]
-
- return
-}
diff --git a/next.config.mjs b/next.config.mjs
index c830d6e..aecfe96 100644
--- a/next.config.mjs
+++ b/next.config.mjs
@@ -1,7 +1,7 @@
/** @type {import('next').NextConfig} */
const nextConfig = {
reactStrictMode: true,
- output: 'standalone',
+ output: 'export',
trailingSlash: true // ensure pages get their own directory in output
}
diff --git a/package.json b/package.json
index 4caf3da..6f9d38a 100644
--- a/package.json
+++ b/package.json
@@ -14,7 +14,7 @@
"format": "npx standard --fix",
"prepare": "husky",
"deploy": "./util/pre-deploy.sh && ./util/deploy-gcloud.sh",
- "test": "jest --verbose",
+ "test": "jest --verbose --passWithNoTests",
"lint": "next lint"
},
"dependencies": {
diff --git a/public/sitemap-0.xml b/public/sitemap-0.xml
index 43f1faf..123b843 100644
--- a/public/sitemap-0.xml
+++ b/public/sitemap-0.xml
@@ -1,10 +1,11 @@
-https://www.aaronjy.me/2025-03-09T12:11:53.510Zweekly0.7
-https://www.aaronjy.me/cv/2025-03-09T12:11:53.511Zweekly0.7
-https://www.aaronjy.me/writing/2025-03-09T12:11:53.511Zweekly0.7
-https://www.aaronjy.me/writing/performance-considerations-tcp-game-server/2025-03-09T12:11:53.511Zweekly0.7
-https://www.aaronjy.me/writing/quick-reflection-katherine-may/2025-03-09T12:11:53.511Zweekly0.7
-https://www.aaronjy.me/writing/static-site-on-google-cloud/2025-03-09T12:11:53.511Zweekly0.7
-https://www.aaronjy.me/writing/support-content-filte-structure-changes-on-a-static-site/2025-03-09T12:11:53.511Zweekly0.7
+https://www.aaronjy.me/2025-03-09T18:28:59.153Zweekly0.7
+https://www.aaronjy.me/about/2025-03-09T18:28:59.153Zweekly0.7
+https://www.aaronjy.me/cv/2025-03-09T18:28:59.153Zweekly0.7
+https://www.aaronjy.me/writing/2025-03-09T18:28:59.153Zweekly0.7
+https://www.aaronjy.me/writing/performance-considerations-tcp-game-server/2025-03-09T18:28:59.153Zweekly0.7
+https://www.aaronjy.me/writing/quick-reflection-katherine-may/2025-03-09T18:28:59.153Zweekly0.7
+https://www.aaronjy.me/writing/static-site-on-google-cloud/2025-03-09T18:28:59.153Zweekly0.7
+https://www.aaronjy.me/writing/support-content-filte-structure-changes-on-a-static-site/2025-03-09T18:28:59.153Zweekly0.7
\ No newline at end of file