Set up manual compatibility tests (#171)
This commit is contained in:
parent
e28a084dbb
commit
d36b336fdd
40 changed files with 1862 additions and 13286 deletions
39
compatibility-tests/scripts/build-setups.mjs
Normal file
39
compatibility-tests/scripts/build-setups.mjs
Normal file
|
@ -0,0 +1,39 @@
|
|||
/**
|
||||
* Build the test setups
|
||||
*/
|
||||
|
||||
import path from 'path'
|
||||
import {colorize, getCompatibilityTestSetups} from './utils.mjs'
|
||||
|
||||
const root = path.resolve(__dirname, '../..')
|
||||
const absPathOfCompatibilityTestSetups = getCompatibilityTestSetups(root)
|
||||
|
||||
const setupsWithErros = []
|
||||
|
||||
// Try building the setups
|
||||
;(async function () {
|
||||
for (const setupDir of absPathOfCompatibilityTestSetups) {
|
||||
try {
|
||||
cd(setupDir)
|
||||
const pathToSetup = path.join(absPathOfCompatibilityTestSetups, setupDir)
|
||||
fs.removeSync(path.join(pathToSetup, 'node_modules'))
|
||||
fs.removeSync(path.join(pathToSetup, 'package-lock.json'))
|
||||
fs.removeSync(path.join(pathToSetup, 'yarn.lock'))
|
||||
await $`npm install`
|
||||
await $`npm run build`
|
||||
} catch (err) {
|
||||
console.error(err)
|
||||
setupsWithErros.push(setupDir)
|
||||
}
|
||||
}
|
||||
|
||||
// Stop if there were any errors during the build process,
|
||||
// and print all of them to the console.
|
||||
if (setupsWithErros.length !== 0) {
|
||||
throw new Error(
|
||||
`The following setups had problems when their dependencies were being installed:\n${colorize.red(
|
||||
setupsWithErros.join('\n'),
|
||||
)}`,
|
||||
)
|
||||
}
|
||||
})()
|
5
compatibility-tests/scripts/ci.mjs
Normal file
5
compatibility-tests/scripts/ci.mjs
Normal file
|
@ -0,0 +1,5 @@
|
|||
import {startRegistry} from './utils.mjs'
|
||||
;(async function runCI() {
|
||||
await startRegistry()
|
||||
process.exit(0)
|
||||
})()
|
3
compatibility-tests/scripts/start-registry.mjs
Normal file
3
compatibility-tests/scripts/start-registry.mjs
Normal file
|
@ -0,0 +1,3 @@
|
|||
import {startRegistry} from './utils.mjs'
|
||||
|
||||
startRegistry()
|
261
compatibility-tests/scripts/utils.mjs
Normal file
261
compatibility-tests/scripts/utils.mjs
Normal file
|
@ -0,0 +1,261 @@
|
|||
/**
|
||||
* Utility functions for the compatibility tests
|
||||
*/
|
||||
|
||||
import fs from 'fs'
|
||||
import path from 'path'
|
||||
import {YAML} from 'zx'
|
||||
import onCleanup from 'node-cleanup'
|
||||
import * as verdaccioPackage from 'verdaccio'
|
||||
|
||||
// 'verdaccio' is not an es module so we have to do this:
|
||||
const startVerdaccioServer = verdaccioPackage.default.default
|
||||
|
||||
export const VERDACCIO_PORT = 4823
|
||||
export const VERDACCIO_HOST = `localhost`
|
||||
export const VERDACCIO_URL = `http://${VERDACCIO_HOST}:${VERDACCIO_PORT}/`
|
||||
export const MONOREPO_ROOT = path.join(__dirname, '../..')
|
||||
export const PATH_TO_YARNRC = path.join(MONOREPO_ROOT, '.yarnrc.yml')
|
||||
|
||||
/**
|
||||
* This script will:
|
||||
* 1. Start verdaccio (a local npm registry),
|
||||
* 2. Configure npm (and _not_ yarn) to use verdaccio as its registry
|
||||
* 3. It will _not_ affect the global yarn installation yet. That is a TODO
|
||||
* 4. It does _not_ affect the monorepo yarnrc file.
|
||||
*
|
||||
* If the script is interrupted, it'll attempt to restore the npm/yarn
|
||||
* registry config to its original state, but that's not guaranteed.
|
||||
*/
|
||||
export async function startRegistry() {
|
||||
const npmOriginalRegistry = (
|
||||
await $`npm config get registry --location=global`
|
||||
).stdout.trim()
|
||||
onCleanup((exitCode, signal) => {
|
||||
onCleanup.uninstall()
|
||||
$`npm config set registry ${npmOriginalRegistry} --location=global`.then(
|
||||
() => {
|
||||
process.kill(process.pid, signal)
|
||||
},
|
||||
)
|
||||
return false
|
||||
})
|
||||
|
||||
await $`echo "Setting npm registry url to verdaccio's"`
|
||||
await $`npm config set registry ${VERDACCIO_URL} --location=global`
|
||||
|
||||
await $`echo Running verdaccio on ${VERDACCIO_URL}`
|
||||
const verdaccioServer = await startVerdaccio(VERDACCIO_PORT)
|
||||
|
||||
await releaseToVerdaccio()
|
||||
}
|
||||
|
||||
/**
|
||||
* Starts the verdaccio server and returns a promise that resolves when the serve is up and ready
|
||||
*
|
||||
* Credid: https://github.com/storybookjs/storybook/blob/92b23c080d03433765cbc7a60553d036a612a501/scripts/run-registry.ts
|
||||
*/
|
||||
const startVerdaccio = (port) => {
|
||||
let resolved = false
|
||||
return Promise.race([
|
||||
new Promise((resolve) => {
|
||||
const config = {
|
||||
...YAML.parse(
|
||||
fs.readFileSync(path.join(__dirname, '../verdaccio.yml'), 'utf8'),
|
||||
),
|
||||
}
|
||||
|
||||
const onReady = (webServer) => {
|
||||
webServer.listen(port, () => {
|
||||
resolved = true
|
||||
resolve(webServer)
|
||||
})
|
||||
}
|
||||
|
||||
startVerdaccioServer(
|
||||
config,
|
||||
6000,
|
||||
undefined,
|
||||
'1.0.0',
|
||||
'verdaccio',
|
||||
onReady,
|
||||
)
|
||||
}),
|
||||
new Promise((_, rej) => {
|
||||
setTimeout(() => {
|
||||
if (!resolved) {
|
||||
resolved = true
|
||||
rej(new Error(`TIMEOUT - verdaccio didn't start within 10s`))
|
||||
}
|
||||
}, 10000)
|
||||
}),
|
||||
])
|
||||
}
|
||||
|
||||
const packagesToPublish = [
|
||||
'@theatre/core',
|
||||
'@theatre/studio',
|
||||
'@theatre/dataverse',
|
||||
'@theatre/react',
|
||||
'@theatre/browser-bundles',
|
||||
'@theatre/r3f',
|
||||
]
|
||||
|
||||
/**
|
||||
* Assigns a new version to each of @theatre/* packages. If there a package depends on another package in this monorepo,
|
||||
* this function makes sure the dependency version is fixed at "hash"
|
||||
*
|
||||
* @param {{name: string, location: string}[]} workspacesListObjects - An Array of objects containing information about the workspaces
|
||||
* @param {string} hash - Hash of the latest commit (or any other string)
|
||||
* @returns {Promise<() => void>} - An async function that restores the package.json files to their original version
|
||||
*/
|
||||
async function assignVersions(workspacesListObjects, hash) {
|
||||
/**
|
||||
* An array of functions each of which restores a certain package.json to its original state
|
||||
* @type {Array<() => void>}
|
||||
*/
|
||||
const restores = []
|
||||
for (const workspaceData of workspacesListObjects) {
|
||||
const pathToPackage = path.resolve(
|
||||
MONOREPO_ROOT,
|
||||
workspaceData.location,
|
||||
'./package.json',
|
||||
)
|
||||
|
||||
const originalFileContent = fs.readFileSync(pathToPackage, {
|
||||
encoding: 'utf-8',
|
||||
})
|
||||
const originalJson = JSON.parse(originalFileContent)
|
||||
|
||||
restores.push(() => {
|
||||
fs.writeFileSync(pathToPackage, originalFileContent, {encoding: 'utf-8'})
|
||||
})
|
||||
|
||||
let {dependencies, peerDependencies, devDependencies} = originalJson
|
||||
const version = hash
|
||||
|
||||
// Normally we don't have to override the package versions in dependencies because yarn would already convert
|
||||
// all the "workspace:*" versions to a fixed version before publishing. However, packages like @theatre/studio
|
||||
// have a peerDependency on @theatre/core set to "*" (meaning they would work with any version of @theatre/core).
|
||||
// This is not the desired behavior in pre-release versions, so here, we'll fix those "*" versions to the set version.
|
||||
for (const deps of [dependencies, peerDependencies, devDependencies]) {
|
||||
if (!deps) continue
|
||||
for (const wpObject of workspacesListObjects) {
|
||||
if (deps[wpObject.name]) {
|
||||
deps[wpObject.name] = hash
|
||||
}
|
||||
}
|
||||
}
|
||||
const newJson = {
|
||||
...originalJson,
|
||||
version,
|
||||
dependencies,
|
||||
peerDependencies,
|
||||
devDependencies,
|
||||
}
|
||||
fs.writeFileSync(pathToPackage, JSON.stringify(newJson, undefined, 2), {
|
||||
encoding: 'utf-8',
|
||||
})
|
||||
}
|
||||
return () =>
|
||||
restores.forEach((fn) => {
|
||||
fn()
|
||||
})
|
||||
}
|
||||
|
||||
/**
|
||||
* Builds all the @theatre/* packages with version number 0.0.1-COMPAT.1 and publishes
|
||||
* them all to the verdaccio registry
|
||||
*/
|
||||
async function releaseToVerdaccio() {
|
||||
const version = '0.0.1-COMPAT.1'
|
||||
cd(MONOREPO_ROOT)
|
||||
|
||||
// @ts-ignore ignore
|
||||
process.env.THEATRE_IS_PUBLISHING = true
|
||||
|
||||
const workspacesListString = await $`yarn workspaces list --json`
|
||||
const workspacesListObjects = workspacesListString.stdout
|
||||
.split(os.EOL)
|
||||
// strip out empty lines
|
||||
.filter(Boolean)
|
||||
.map((x) => JSON.parse(x))
|
||||
|
||||
const restorePackages = await assignVersions(workspacesListObjects, version)
|
||||
|
||||
process.on('SIGINT', async function cleanup(a) {
|
||||
restorePackages()
|
||||
process.exit(0)
|
||||
})
|
||||
|
||||
// set verdaccio as the publish registry, and add it to the whitelist
|
||||
const restoreYarnRc = patchYarnRcToUseVerdaccio()
|
||||
|
||||
await $`yarn clean`
|
||||
await $`yarn build`
|
||||
|
||||
await Promise.all(
|
||||
packagesToPublish.map(async (workspaceName) => {
|
||||
const npmTag = 'compat'
|
||||
await $`yarn workspace ${workspaceName} npm publish --access public --tag ${npmTag}`
|
||||
}),
|
||||
)
|
||||
|
||||
restorePackages()
|
||||
restoreYarnRc()
|
||||
}
|
||||
|
||||
/**
|
||||
* Temporarily patches the yarnrc file to sue verdaccio as its publish registry.
|
||||
*
|
||||
* Restores yarnrc to the old version when restoreYarnRc() is called.
|
||||
*/
|
||||
function patchYarnRcToUseVerdaccio() {
|
||||
const originalYarnrcContent = fs.readFileSync(PATH_TO_YARNRC, {
|
||||
encoding: 'utf-8',
|
||||
})
|
||||
|
||||
const newYarnRcContent = YAML.stringify({
|
||||
...YAML.parse(originalYarnrcContent),
|
||||
unsafeHttpWhitelist: [VERDACCIO_HOST],
|
||||
npmPublishRegistry: VERDACCIO_URL,
|
||||
npmAuthIdent: 'test:test',
|
||||
})
|
||||
|
||||
fs.writeFileSync(PATH_TO_YARNRC, newYarnRcContent, {encoding: 'utf-8'})
|
||||
|
||||
return function restoreYarnRc() {
|
||||
fs.writeFileSync(PATH_TO_YARNRC, originalYarnrcContent, {encoding: 'utf-8'})
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get all the setups from `./compatibility-tests/`
|
||||
*
|
||||
* @returns {Array<string>} An array containing the absolute paths to the compatibility test setups
|
||||
*/
|
||||
export function getCompatibilityTestSetups() {
|
||||
const buildTestsDir = path.join(MONOREPO_ROOT, 'compatibility-tests')
|
||||
let buildTestsDirEntries
|
||||
|
||||
try {
|
||||
buildTestsDirEntries = fs.readdirSync(buildTestsDir)
|
||||
} catch {
|
||||
throw new Error(
|
||||
`Could not list directory: "${buildTestsDir}" Is it an existing directory?`,
|
||||
)
|
||||
}
|
||||
const setupsAbsPaths = []
|
||||
|
||||
// NOTE: We assume that every directory matching `compatibility-tests/test-*` is
|
||||
// a test package
|
||||
for (const entry of buildTestsDirEntries) {
|
||||
if (!entry.startsWith('test-')) continue
|
||||
const entryAbsPath = path.join(buildTestsDir, entry)
|
||||
if (fs.lstatSync(entryAbsPath).isDirectory()) {
|
||||
setupsAbsPaths.push(entryAbsPath)
|
||||
}
|
||||
}
|
||||
|
||||
return setupsAbsPaths
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue