Skip to content

Instantly share code, notes, and snippets.

@prescottprue
Last active November 27, 2024 02:14
Show Gist options
  • Save prescottprue/e8f2a437e647285340dea50a495a811b to your computer and use it in GitHub Desktop.
Save prescottprue/e8f2a437e647285340dea50a495a811b to your computer and use it in GitHub Desktop.
Replace Jest With Vitest
#!/bin/bash
# Based off https://gist.github.com/wojtekmaj/6defa1f358daae28bd52b7b6dbeb7ab6 with a few fixes and
# additions/removals including no git interactions
join_by() {
local d=${1-} f=${2-}
if shift 2; then
printf %s "$f" "${@/#/$d}"
fi
}
# Define functions
add_vitest_config() {
setupFile=""
if [ -f "./jest.setup.js" ]; then
mv jest.setup.js vitest.setup.ts
setupFile="
test: {
setupFiles: 'vitest.setup.ts',
},"
fi
# Add Vitest config
echo "import { defineConfig } from 'vitest/config';
export default defineConfig({$setupFile
coverage: {
reporter: ['lcov', 'json'],
include: ['src/**'],
},
});" > vitest.config.ts
rm jest.config.js
}
add_vitest_setup() {
# Copy existing jest setup into Vitest setup
if [ -f "./jest.setup.js" ]; then
echo "Migrating jest.setup.js file"
mv jest.setup.js vitest.setup.ts
fi
}
add_vitest_script() {
# Replace test command in package.json
npm pkg delete scripts.test
npm pkg set scripts.test="vitest"
}
# Add vitest and coverage reporter
yarn add -D --exact vitest @vitest/coverage-v8
# Remove jest and any associated packages
packagesToRemove=jest
if grep -q "@types/jest" package.json; then
packagesToRemove="${packagesToRemove} @types/jest"
fi
if grep -q "@swc/jest" package.json; then
packagesToRemove="${packagesToRemove} @swc/jest"
fi
yarn remove $packagesToRemove
# Dedupe dependencies
yarn dedupe
# Add vitest config file and remove jest config
add_vitest_config
# Add Vitest setup
add_vitest_setup
# Add Vitest script to npm scripts
add_vitest_script
# Go to src directory
cd src
# Find all *.spec.js, *.spec.ts, *.spec.jsx, and *.spec.tsx files. For each of them:
# - Replace "jest.*" with "vi.*"
# - Attempt to replace "jest.mock" and "jest.requireActual" with "vi.mock" and "vi.importActual", respectively.
# - Add "import { describe, expect, it, … } from 'vitest';" to the top of the file
files=$(find . -type f -name "*.spec.js" -o -name "*.spec.ts" -o -name "*.spec.jsx" -o -name "*.spec.tsx" -o -name "*.test.js" -o -name "*.test.ts" -o -name "*.test.jsx" -o -name "*.test.tsx" | grep -v "node_modules")
for file in $files; do
echo "Processing $file"
# Detect if file contains "from ["']vitest["']". If so, skip it.
if grep -q "from ['\']vitest['\']" $file; then
echo " Test file appears to have already been migrated to Vitest. Skipping"
continue
fi
# Detect if file contains "from ["']@playwright/test["']". If so, skip it.
if grep -q "from ['\"]@playwright\/test['\"]" $file; then
echo " Test file appears to be using Playwright. Skipping"
continue
fi
# Replace "jest.clearAllMocks" with "vi.clearAllMocks"
sed -i '' 's/jest.clearAllMocks/vi.clearAllMocks/g' $file
# Replace "jest.fn" with "vi.fn"
sed -i '' 's/jest.fn/vi.fn/g' $file
# Replace "jest.mocked" with "vi.mocked"
sed -i '' 's/jest.mocked/vi.mocked/g' $file
# Replace "jest.resetAllMocks" with "vi.resetAllMocks"
sed -i '' 's/jest.resetAllMocks/vi.resetAllMocks/g' $file
# Replace "jest.restoreAllMocks" with "vi.restoreAllMocks"
sed -i '' 's/jest.restoreAllMocks/vi.restoreAllMocks/g' $file
# Replace "jest.resetModules" with "vi.resetModules"
sed -i '' 's/jest.resetModules/vi.resetModules/g' $file
# Replace "jest.spyOn" with "vi.spyOn"
sed -i '' 's/jest.spyOn/vi.spyOn/g' $file
# Replace "jest.useFakeTimers" with "vi.useFakeTimers"
sed -i '' 's/jest.useFakeTimers/vi.useFakeTimers/g' $file
# Replace "jest.useRealTimers" with "vi.useRealTimers"
sed -i '' 's/jest.useRealTimers/vi.useRealTimers/g' $file
# Replace "jest.SpyInstance" with "MockInstance"
sed -i '' 's/jest.SpyInstance/MockInstance/g' $file
# Replace "advanceTimers: jest.advanceTimersByTime" with "advanceTimers: vi.advanceTimersByTime.bind(vi)"
sed -i '' 's/advanceTimers: jest.advanceTimersByTime/advanceTimers: vi.advanceTimersByTime.bind(vi)/g' $file
# Detect jest.mock(). Since vi.mock() uses ESM modules, chances are manual changes
# are going to be necessary. So, we'll print a warning.
if grep -q "jest.mock(" $file; then
echo " Warning: $file contained jest.mock(). You'll likely need to manually fix vi.mock() implementation."
sed -i '' 's/jest.mock/vi.mock/g' $file
fi
# Detect jest.requireActual(). Since vi.importActual() uses ESM modules, chances are
# manual changes are going to be necessary. So, we'll print a warning.
if grep -q "jest.requireActual(" $file; then
echo " Warning: $file contained jest.requireActual(). You'll likely need to manually fix vi.importActual() implementation."
sed -i '' 's/jest.requireActual/vi.importActual/g' $file
fi
# Replace jest.setTimeout() with vi.setConfig({ testTimeout: N })
if grep -q "jest.setTimeout(" $file; then
echo " TODO"
fi
# Replace any remaining instance of jest with vi (helps jest.* with methods on newline and with tabs)
sed -i '' 's/jest/vi/g' $file
# Define list of required imports for each file
# Clear the array of imports
imports=()
# If file contains "afterEach", add it to the list of required imports
if grep -q "afterEach" $file; then
imports[0]="afterEach"
fi
# If file contains "beforeEach", add it to the list of required imports
if grep -q "beforeEach" $file; then
imports[1]="beforeEach"
fi
imports[2]="describe"
imports[3]="expect"
imports[4]="it"
# If file contains "vi.", add it to the list of required imports
if grep -q "vi\." $file; then
imports[5]="vi"
fi
if grep -q "beforeAll" $file; then
imports[6]="beforeAll"
fi
if grep -q "afterAll" $file; then
imports[7]="afterAll"
fi
if grep -q "test\." $file; then
imports[8]="test"
fi
if grep -q "MockInstance" $file; then
imports[8]="type MockInstance"
fi
if grep -q "vi\.Mock" $file; then
imports[8]="type Mock"
fi
# Wait to replace vi.Mock (from jest.Mock) so that Mock type import can be more accuratley detected
sed -i '' 's/vi.Mock/Mock/g' $file
# Join list of required imports into a string
imports_string=$(join_by ", " "${imports[@]}")
# Add "import { … } from 'vitest';" to the top of the file.
sed -i '' '1i\
import { '"$imports_string"' } from "vitest";
' $file
echo " Done"
done
# Return to root directory
cd ../
# Run prettier formatting if prettier exists
if grep -q "prettier" package.json; then
yarn prettier --write .
fi
# Run lint to be sure everything is working, exit if not
yarn lint || exit 1
# Run tests to be sure everything is working, exit if not
yarn test || exit 1
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment