mirror of
https://github.com/apache/superset.git
synced 2026-05-21 15:55:10 +00:00
396 lines
15 KiB
JavaScript
396 lines
15 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
/**
|
|
* Licensed to the Apache Software Foundation (ASF) under one
|
|
* or more contributor license agreements. See the NOTICE file
|
|
* distributed with this work for additional information
|
|
* regarding copyright ownership. The ASF licenses this file
|
|
* to you under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance
|
|
* with the License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing,
|
|
* software distributed under the License is distributed on an
|
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
* KIND, either express or implied. See the License for the
|
|
* specific language governing permissions and limitations
|
|
* under the License.
|
|
*/
|
|
|
|
import fs from 'fs';
|
|
import path from 'path';
|
|
import { execSync } from 'child_process';
|
|
import { fileURLToPath } from 'url';
|
|
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = path.dirname(__filename);
|
|
|
|
const CONFIG_FILE = path.join(__dirname, '..', 'versions-config.json');
|
|
|
|
// Parse command line arguments
|
|
const rawArgs = process.argv.slice(2);
|
|
const skipGenerate = rawArgs.includes('--skip-generate');
|
|
const args = rawArgs.filter((a) => a !== '--skip-generate');
|
|
const command = args[0]; // 'add' or 'remove'
|
|
const section = args[1]; // 'user_docs', 'admin_docs', 'developer_docs', or 'components'
|
|
const version = args[2]; // version string like '1.2.0'
|
|
|
|
function loadConfig() {
|
|
return JSON.parse(fs.readFileSync(CONFIG_FILE, 'utf8'));
|
|
}
|
|
|
|
function saveConfig(config) {
|
|
fs.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2) + '\n');
|
|
}
|
|
|
|
function freezeDataImports(section, version) {
|
|
// MDX files can `import` JSON/YAML data from outside the section, either
|
|
// via escaping relative paths (e.g. country-map-tools.mdx imports
|
|
// `../../data/countries.json`) or via the `@site/` alias (e.g.
|
|
// feature-flags.mdx imports `@site/static/feature-flags.json`). Without
|
|
// intervention the snapshot keeps reading the live file, so the
|
|
// historical version's content silently changes whenever the data file
|
|
// is updated. Copy each escaping data import into a snapshot-local
|
|
// `_versioned_data/` dir and rewrite the import to point there.
|
|
// The user_docs section's source content lives in `docs/docs/` (the
|
|
// historical folder name), while admin_docs / developer_docs /
|
|
// components match their plugin id 1:1.
|
|
const sectionDir = section === 'user_docs' ? 'docs' : section;
|
|
const sectionRoot = path.join(__dirname, '..', sectionDir);
|
|
const docsRoot = path.join(__dirname, '..');
|
|
const versionedDocsDir = `${section}_versioned_docs/version-${version}`;
|
|
const versionedDocsPath = path.join(__dirname, '..', versionedDocsDir);
|
|
const frozenDataDir = path.join(versionedDocsPath, '_versioned_data');
|
|
|
|
if (!fs.existsSync(versionedDocsPath)) {
|
|
return;
|
|
}
|
|
|
|
console.log(` Freezing data imports in ${versionedDocsDir}...`);
|
|
|
|
// Matches data file imports in two flavors:
|
|
// `from '../../foo/bar.json'` (relative, must escape one or more dirs)
|
|
// `from '@site/static/foo.json'` (Docusaurus site-root alias)
|
|
const dataImportRe = /(from\s+['"])((?:\.\.\/)+|@site\/)([^'"\s]+\.(?:json|ya?ml))(['"])/g;
|
|
|
|
function freezeOne(fullPath, depth, prefix, pathSpec, importPath, suffix) {
|
|
let resolvedSource;
|
|
if (pathSpec === '@site/') {
|
|
// `@site/...` always resolves relative to the docs root.
|
|
resolvedSource = path.join(docsRoot, importPath);
|
|
} else {
|
|
// Relative path — must escape the file's depth within the section
|
|
// to point at content outside the section. Imports that stay inside
|
|
// are copied wholesale by Docusaurus, so we leave them alone.
|
|
const upCount = pathSpec.match(/\.\.\//g).length;
|
|
if (upCount <= depth) return null;
|
|
const relativeFromVersioned = path.relative(versionedDocsPath, fullPath);
|
|
const originalDir = path.dirname(path.join(sectionRoot, relativeFromVersioned));
|
|
resolvedSource = path.resolve(originalDir, pathSpec + importPath);
|
|
}
|
|
// Skip imports that land inside the section root — those get copied
|
|
// with the section snapshot already.
|
|
const relFromSection = path.relative(sectionRoot, resolvedSource);
|
|
if (!relFromSection.startsWith('..')) return null;
|
|
const relFromDocsRoot = path.relative(docsRoot, resolvedSource);
|
|
if (relFromDocsRoot.startsWith('..') || !fs.existsSync(resolvedSource)) {
|
|
return null;
|
|
}
|
|
const destPath = path.join(frozenDataDir, relFromDocsRoot);
|
|
fs.mkdirSync(path.dirname(destPath), { recursive: true });
|
|
fs.copyFileSync(resolvedSource, destPath);
|
|
const rewritten = path
|
|
.relative(path.dirname(fullPath), destPath)
|
|
.split(path.sep)
|
|
.join('/');
|
|
const finalImport = rewritten.startsWith('.') ? rewritten : `./${rewritten}`;
|
|
return `${prefix}${finalImport}${suffix}`;
|
|
}
|
|
|
|
function walk(dir, depth) {
|
|
for (const entry of fs.readdirSync(dir, { withFileTypes: true })) {
|
|
const fullPath = path.join(dir, entry.name);
|
|
if (entry.isDirectory()) {
|
|
if (entry.name.startsWith('_')) continue;
|
|
walk(fullPath, depth + 1);
|
|
} else if (entry.isFile() && /\.(md|mdx)$/.test(entry.name)) {
|
|
const original = fs.readFileSync(fullPath, 'utf8');
|
|
let inFence = false;
|
|
let mutated = false;
|
|
const updated = original.split('\n').map(line => {
|
|
if (/^\s*(```|~~~)/.test(line)) {
|
|
inFence = !inFence;
|
|
return line;
|
|
}
|
|
if (inFence) return line;
|
|
return line.replace(dataImportRe, (match, prefix, pathSpec, importPath, suffix) => {
|
|
const rewritten = freezeOne(fullPath, depth, prefix, pathSpec, importPath, suffix);
|
|
if (rewritten === null) return match;
|
|
mutated = true;
|
|
return rewritten;
|
|
});
|
|
}).join('\n');
|
|
if (mutated) {
|
|
fs.writeFileSync(fullPath, updated);
|
|
const rel = path.relative(versionedDocsPath, fullPath);
|
|
console.log(` Froze data imports in ${rel}`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
walk(versionedDocsPath, 0);
|
|
}
|
|
|
|
function fixVersionedImports(section, version) {
|
|
// Versioned content lands one directory deeper than the source content,
|
|
// so any `../../src/` or `../../data/` imports in .md/.mdx files need
|
|
// an extra `../` to keep reaching docs/src and docs/data.
|
|
const versionedDocsDir = `${section}_versioned_docs/version-${version}`;
|
|
const versionedDocsPath = path.join(__dirname, '..', versionedDocsDir);
|
|
|
|
if (!fs.existsSync(versionedDocsPath)) {
|
|
return;
|
|
}
|
|
|
|
console.log(` Fixing relative imports in ${versionedDocsDir}...`);
|
|
|
|
// Imports whose `../` count exceeds the file's depth within the section
|
|
// escape the section root, so they need one extra `../` once the file
|
|
// lives one level deeper inside the snapshot dir. Imports that stay
|
|
// inside the section are unaffected (the section copies wholesale).
|
|
function walk(dir, depth) {
|
|
for (const entry of fs.readdirSync(dir, { withFileTypes: true })) {
|
|
const fullPath = path.join(dir, entry.name);
|
|
if (entry.isDirectory()) {
|
|
walk(fullPath, depth + 1);
|
|
} else if (entry.isFile() && /\.(md|mdx)$/.test(entry.name)) {
|
|
const original = fs.readFileSync(fullPath, 'utf8');
|
|
// Track fenced code blocks so we don't rewrite import samples inside
|
|
// ```ts / ```js (etc.) blocks that are documentation, not real imports.
|
|
let inFence = false;
|
|
const updated = original.split('\n').map(line => {
|
|
if (/^\s*(```|~~~)/.test(line)) {
|
|
inFence = !inFence;
|
|
return line;
|
|
}
|
|
if (inFence) return line;
|
|
return line.replace(
|
|
/(from\s+['"])((?:\.\.\/)+)/g,
|
|
(match, prefix, dots) => {
|
|
const upCount = dots.match(/\.\.\//g).length;
|
|
return upCount > depth ? `${prefix}../${dots}` : match;
|
|
},
|
|
);
|
|
}).join('\n');
|
|
if (updated !== original) {
|
|
fs.writeFileSync(fullPath, updated);
|
|
const rel = path.relative(versionedDocsPath, fullPath);
|
|
console.log(` Fixed imports in ${rel}`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
walk(versionedDocsPath, 0);
|
|
}
|
|
|
|
function addVersion(section, version) {
|
|
const config = loadConfig();
|
|
|
|
if (!config[section]) {
|
|
console.error(`Section '${section}' not found in config`);
|
|
process.exit(1);
|
|
}
|
|
|
|
// Check if version already exists
|
|
if (config[section].onlyIncludeVersions.includes(version)) {
|
|
console.error(`Version ${version} already exists in ${section}`);
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log(`Creating version ${version} for ${section}...`);
|
|
|
|
// Refresh auto-generated content (database pages, API reference,
|
|
// component playground) so the snapshot captures the current state of
|
|
// master rather than whatever happened to be on disk. `generate:smart`
|
|
// hashes its inputs and skips unchanged generators, so this is cheap
|
|
// when the dev already has fresh output.
|
|
//
|
|
// Use --skip-generate if you've placed a CI-artifact databases.json
|
|
// (the `database-diagnostics` artifact from Python-Integration) and
|
|
// want to preserve it instead of letting the local env regenerate it.
|
|
// See docs/README.md "Before You Cut" for the canonical release flow.
|
|
if (skipGenerate) {
|
|
console.log(` Skipping auto-gen refresh (--skip-generate set)`);
|
|
} else {
|
|
console.log(` Refreshing auto-generated docs...`);
|
|
try {
|
|
execSync('yarn run generate:smart', { stdio: 'inherit' });
|
|
} catch (error) {
|
|
console.error(`Failed to refresh auto-generated docs: ${error.message}`);
|
|
process.exit(1);
|
|
}
|
|
}
|
|
|
|
// Run Docusaurus version command
|
|
const docusaurusCommand = `yarn docusaurus docs:version:${section} ${version}`;
|
|
|
|
try {
|
|
execSync(docusaurusCommand, { stdio: 'inherit' });
|
|
} catch (error) {
|
|
console.error(`Failed to create version: ${error.message}`);
|
|
process.exit(1);
|
|
}
|
|
|
|
// Freeze data imports BEFORE adjusting paths, so the depth-aware rewriter
|
|
// doesn't process the now-local imports we just rewrote.
|
|
freezeDataImports(section, version);
|
|
|
|
// Fix relative imports in versioned content
|
|
fixVersionedImports(section, version);
|
|
|
|
// Update config
|
|
// Add to onlyIncludeVersions array (after 'current')
|
|
const versionIndex = config[section].onlyIncludeVersions.indexOf('current') + 1;
|
|
config[section].onlyIncludeVersions.splice(versionIndex, 0, version);
|
|
|
|
// Add version metadata
|
|
config[section].versions[version] = {
|
|
label: version,
|
|
path: version,
|
|
banner: 'none'
|
|
};
|
|
|
|
// Note: we deliberately do NOT auto-bump `lastVersion` to the new
|
|
// version. Superset's docs site keeps `lastVersion: 'current'` so
|
|
// the canonical URLs (`/user-docs/...`, `/admin-docs/...`,
|
|
// `/developer-docs/...`, `/components/...`) always render master
|
|
// content; cut versions are accessed only via their explicit version
|
|
// segment. (`/docs/...` paths are legacy and handled via per-page
|
|
// redirects in docusaurus.config.ts — not a current canonical
|
|
// form.) If you want a different policy, edit versions-config.json
|
|
// after cutting.
|
|
|
|
saveConfig(config);
|
|
console.log(`✅ Version ${version} added successfully to ${section}`);
|
|
console.log(`📝 Updated versions-config.json`);
|
|
}
|
|
|
|
function removeVersion(section, version) {
|
|
const config = loadConfig();
|
|
|
|
if (!config[section]) {
|
|
console.error(`Section '${section}' not found in config`);
|
|
process.exit(1);
|
|
}
|
|
|
|
if (version === 'current') {
|
|
console.error(`Cannot remove 'current' version`);
|
|
process.exit(1);
|
|
}
|
|
|
|
if (!config[section].onlyIncludeVersions.includes(version)) {
|
|
console.error(`Version ${version} not found in ${section}`);
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log(`Removing version ${version} from ${section}...`);
|
|
|
|
// Determine file paths based on section
|
|
const versionedDocsDir = `${section}_versioned_docs/version-${version}`;
|
|
const versionedSidebarsFile = `${section}_versioned_sidebars/version-${version}-sidebars.json`;
|
|
|
|
// Remove versioned files
|
|
const docsPath = path.join(__dirname, '..', versionedDocsDir);
|
|
const sidebarsPath = path.join(__dirname, '..', versionedSidebarsFile);
|
|
|
|
if (fs.existsSync(docsPath)) {
|
|
fs.rmSync(docsPath, { recursive: true });
|
|
console.log(` Removed ${versionedDocsDir}`);
|
|
}
|
|
|
|
if (fs.existsSync(sidebarsPath)) {
|
|
fs.unlinkSync(sidebarsPath);
|
|
console.log(` Removed ${versionedSidebarsFile}`);
|
|
}
|
|
|
|
// Update versions.json file
|
|
const versionsJsonFile = `${section}_versions.json`;
|
|
const versionsJsonPath = path.join(__dirname, '..', versionsJsonFile);
|
|
|
|
if (fs.existsSync(versionsJsonPath)) {
|
|
const versions = JSON.parse(fs.readFileSync(versionsJsonPath, 'utf8'));
|
|
const versionIndex = versions.indexOf(version);
|
|
if (versionIndex > -1) {
|
|
versions.splice(versionIndex, 1);
|
|
if (versions.length === 0) {
|
|
// Sections with no versions shouldn't carry an empty versions file
|
|
// on disk — Docusaurus doesn't require it, and an empty `[]` file
|
|
// gets picked up by `docusaurus version` and snapshotted into the
|
|
// next cut.
|
|
fs.unlinkSync(versionsJsonPath);
|
|
console.log(` Removed empty ${versionsJsonFile}`);
|
|
} else {
|
|
fs.writeFileSync(versionsJsonPath, JSON.stringify(versions, null, 2) + '\n');
|
|
console.log(` Updated ${versionsJsonFile}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Update config
|
|
const versionIndex = config[section].onlyIncludeVersions.indexOf(version);
|
|
config[section].onlyIncludeVersions.splice(versionIndex, 1);
|
|
delete config[section].versions[version];
|
|
|
|
// Update lastVersion if needed
|
|
if (config[section].lastVersion === version) {
|
|
// Set to the next available version or 'current'
|
|
const remainingVersions = config[section].onlyIncludeVersions.filter(v => v !== 'current');
|
|
config[section].lastVersion = remainingVersions.length > 0 ? remainingVersions[0] : 'current';
|
|
console.log(` Updated lastVersion to ${config[section].lastVersion}`);
|
|
}
|
|
|
|
saveConfig(config);
|
|
console.log(`✅ Version ${version} removed successfully from ${section}`);
|
|
console.log(`📝 Updated versions-config.json`);
|
|
}
|
|
|
|
function printUsage() {
|
|
console.log(`
|
|
Usage:
|
|
node scripts/manage-versions.mjs add <section> <version> [--skip-generate]
|
|
node scripts/manage-versions.mjs remove <section> <version>
|
|
|
|
Where:
|
|
- section: 'user_docs', 'admin_docs', 'developer_docs', or 'components'
|
|
- version: version string (e.g., '1.2.0', '2.0.0')
|
|
- --skip-generate: skip refreshing auto-generated docs before snapshotting
|
|
(use when you've already placed a fresh databases.json
|
|
from CI and want to preserve it)
|
|
|
|
Examples:
|
|
node scripts/manage-versions.mjs add user_docs 2.0.0
|
|
node scripts/manage-versions.mjs add developer_docs 1.3.0
|
|
node scripts/manage-versions.mjs remove components 1.0.0
|
|
`);
|
|
}
|
|
|
|
// Main execution
|
|
if (!command || !section || !version) {
|
|
printUsage();
|
|
process.exit(1);
|
|
}
|
|
|
|
if (command === 'add') {
|
|
addVersion(section, version);
|
|
} else if (command === 'remove') {
|
|
removeVersion(section, version);
|
|
} else {
|
|
console.error(`Unknown command: ${command}`);
|
|
printUsage();
|
|
process.exit(1);
|
|
}
|