mirror of
https://github.com/github/spec-kit.git
synced 2026-07-04 04:45:43 +08:00
* fix(scripts): count subdirectory-only dirs as non-empty in PowerShell
Test-DirHasFiles (the documented PowerShell twin of bash check_dir) tested
non-emptiness with `Get-ChildItem | Where-Object { -not $_.PSIsContainer }`,
counting only top-level FILES and ignoring subdirectories. Bash check_dir
(`-n $(ls -A ...)`) and the PowerShell JSON-path contracts checks
(check-prerequisites.ps1 / setup-tasks.ps1, no PSIsContainer filter) both
count ANY entry. So a contracts/ directory whose only contents are
subdirectories (e.g. contracts/v1/openapi.yaml) was reported present by
bash, by bash JSON, and by PowerShell JSON, but [FAIL]/absent by PowerShell
text mode — the lone outlier.
Drop the PSIsContainer filter so Test-DirHasFiles counts any entry, matching
the other three code paths.
Add bash + PowerShell parity tests asserting a subdir-only contracts/ dir is
reported non-empty in both shells.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* review: accurate non-empty comment + drop doubled test prefix
Address review feedback on Test-DirHasFiles parity fix:
- Reword the common.ps1 comment so it no longer claims exact `ls -A` parity (Get-ChildItem omits hidden entries without -Force); it now points at the in-repo PowerShell JSON contracts checks as the matching reference and keeps the subdir-only-is-non-empty rationale.
- Rename test_test_dir_has_files_ps_... -> test_dir_has_files_ps_... to drop the doubled 'test_' prefix.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* test: assert dir-non-emptiness via stdout marker, not exit code
Address Copilot review: check_dir always exits 0 (it echoes the marker rather than setting an exit code) and Test-DirHasFiles returns a boolean (pwsh still exits 0 when it returns $false), so 'result.returncode == 0' validated nothing. Drop the misleading assertion and rely on the [OK]/checkmark marker in stdout, which is the actual behavioral signal; document why inline.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* fix: keep common.ps1 ASCII-only (PowerShell 5.1 compatibility)
My reworded Test-DirHasFiles comment introduced an em dash (U+2014), which tripped tests/test_ps1_encoding.py::test_ps1_file_is_ascii_only -- .ps1 files must stay ASCII for Windows PowerShell 5.1. Replace it with '--', matching the existing comment style in this file (e.g. the Resolve-SpecifyInitDir docstring).
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
* test: decode dir-parity subprocess output as UTF-8 explicitly
Address Copilot review: check_dir echoes the non-ASCII markers ✓/✗, and subprocess.run with text=True but no encoding decodes via the platform locale (cp1252 on Windows), which can raise UnicodeDecodeError or mangle stdout. Pin encoding='utf-8' on both the bash and PowerShell dir-parity helpers so decoding is deterministic across CI runners.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
---------
Co-authored-by: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
565 lines
23 KiB
PowerShell
565 lines
23 KiB
PowerShell
#!/usr/bin/env pwsh
|
|
# Common PowerShell functions analogous to common.sh
|
|
|
|
# Find repository root by searching upward for .specify directory
|
|
# This is the primary marker for spec-kit projects
|
|
function Find-SpecifyRoot {
|
|
param([string]$StartDir = (Get-Location).Path)
|
|
|
|
# Normalize to absolute path to prevent issues with relative paths
|
|
# Use -LiteralPath to handle paths with wildcard characters ([, ], *, ?)
|
|
$resolved = Resolve-Path -LiteralPath $StartDir -ErrorAction SilentlyContinue
|
|
$current = if ($resolved) { $resolved.Path } else { $null }
|
|
if (-not $current) { return $null }
|
|
|
|
while ($true) {
|
|
if (Test-Path -LiteralPath (Join-Path $current ".specify") -PathType Container) {
|
|
return $current
|
|
}
|
|
$parent = Split-Path $current -Parent
|
|
if ([string]::IsNullOrEmpty($parent) -or $parent -eq $current) {
|
|
return $null
|
|
}
|
|
$current = $parent
|
|
}
|
|
}
|
|
|
|
# Resolve an explicit SPECIFY_INIT_DIR project override (the directory that
|
|
# *contains* .specify/), for non-interactive / CI use -- e.g. running a Spec Kit
|
|
# command against a member project from a monorepo root without cd.
|
|
#
|
|
# Precondition: $env:SPECIFY_INIT_DIR is set. Returns the validated project root,
|
|
# or writes an error and exits 1. Strict by design: the path must exist and
|
|
# contain .specify/, with no silent fallback. (An empty string is falsy, so the
|
|
# caller's `if ($env:SPECIFY_INIT_DIR)` guard treats empty as unset.)
|
|
#
|
|
# This is the single resolver: bundled extensions inherit it by sourcing core
|
|
# (e.g. the git extension's create-new-feature-branch) rather than duplicating it.
|
|
function Resolve-SpecifyInitDir {
|
|
$initDir = $env:SPECIFY_INIT_DIR
|
|
# Normalize: relative paths resolve against the current directory.
|
|
if (-not [System.IO.Path]::IsPathRooted($initDir)) {
|
|
$initDir = Join-Path (Get-Location).Path $initDir
|
|
}
|
|
$resolved = Resolve-Path -LiteralPath $initDir -ErrorAction SilentlyContinue
|
|
# Resolve-Path also succeeds for files, so check the resolved path is a
|
|
# directory; otherwise a file value would slip through to the less accurate
|
|
# "not a Spec Kit project" error below.
|
|
if (-not $resolved -or -not (Test-Path -LiteralPath $resolved.Path -PathType Container)) {
|
|
[Console]::Error.WriteLine("ERROR: SPECIFY_INIT_DIR does not point to an existing directory: $($env:SPECIFY_INIT_DIR)")
|
|
exit 1
|
|
}
|
|
# Resolve-Path echoes back any trailing separator from the input; trim it so
|
|
# the returned root matches the bash resolver, whose `cd && pwd` never yields
|
|
# one. TrimEndingDirectorySeparator is a no-op on a bare root and on a path
|
|
# that already has no trailing separator.
|
|
$initRoot = [System.IO.Path]::TrimEndingDirectorySeparator($resolved.Path)
|
|
if (-not (Test-Path -LiteralPath (Join-Path $initRoot '.specify') -PathType Container)) {
|
|
[Console]::Error.WriteLine("ERROR: SPECIFY_INIT_DIR is not a Spec Kit project (no .specify/ directory): $initRoot")
|
|
exit 1
|
|
}
|
|
return $initRoot
|
|
}
|
|
|
|
# Get repository root, prioritizing .specify directory
|
|
# This prevents using a parent repository when spec-kit is initialized in a subdirectory
|
|
function Get-RepoRoot {
|
|
# Explicit project override wins (see Resolve-SpecifyInitDir).
|
|
if ($env:SPECIFY_INIT_DIR) {
|
|
return (Resolve-SpecifyInitDir)
|
|
}
|
|
|
|
# First, look for .specify directory (spec-kit's own marker)
|
|
$specifyRoot = Find-SpecifyRoot
|
|
if ($specifyRoot) {
|
|
return $specifyRoot
|
|
}
|
|
|
|
# Final fallback to script location
|
|
# Use -LiteralPath to handle paths with wildcard characters
|
|
return (Resolve-Path -LiteralPath (Join-Path $PSScriptRoot "../../..")).Path
|
|
}
|
|
|
|
function Get-CurrentBranch {
|
|
# Return feature name from explicit state only.
|
|
# Feature state is set by SPECIFY_FEATURE (from create-new-feature or
|
|
# the git extension) or implicitly via .specify/feature.json.
|
|
if ($env:SPECIFY_FEATURE) {
|
|
return $env:SPECIFY_FEATURE
|
|
}
|
|
|
|
# No explicit feature set - return empty to signal "unknown".
|
|
return ""
|
|
}
|
|
|
|
|
|
|
|
# Persist a feature_directory value to .specify/feature.json.
|
|
# Writes only when the file is missing or the value differs from what's stored.
|
|
function Save-FeatureJson {
|
|
param(
|
|
[Parameter(Mandatory = $true)][string]$RepoRoot,
|
|
[Parameter(Mandatory = $true)][string]$FeatureDirectory
|
|
)
|
|
|
|
# Strip repo root prefix if the value is absolute and under repo root.
|
|
# Use case-insensitive comparison on Windows only (case-sensitive filesystems elsewhere).
|
|
$prefix = $RepoRoot + [System.IO.Path]::DirectorySeparatorChar
|
|
if ($null -ne $IsWindows) { $onWin = $IsWindows } else { $onWin = $true }
|
|
if ($onWin) {
|
|
$cmp = [System.StringComparison]::OrdinalIgnoreCase
|
|
} else {
|
|
$cmp = [System.StringComparison]::Ordinal
|
|
}
|
|
if ($FeatureDirectory.StartsWith($prefix, $cmp)) {
|
|
$FeatureDirectory = $FeatureDirectory.Substring($prefix.Length)
|
|
}
|
|
|
|
$fjPath = Join-Path (Join-Path $RepoRoot '.specify') 'feature.json'
|
|
|
|
# Read current value and skip write when unchanged
|
|
if (Test-Path -LiteralPath $fjPath -PathType Leaf) {
|
|
try {
|
|
$raw = Get-Content -LiteralPath $fjPath -Raw
|
|
$cfg = $raw | ConvertFrom-Json
|
|
if ($cfg.feature_directory -eq $FeatureDirectory) {
|
|
return
|
|
}
|
|
} catch {
|
|
# File is corrupt or unreadable - overwrite it
|
|
}
|
|
}
|
|
|
|
# Ensure .specify/ directory exists
|
|
$specifyDir = Join-Path $RepoRoot '.specify'
|
|
if (-not (Test-Path -LiteralPath $specifyDir -PathType Container)) {
|
|
New-Item -ItemType Directory -Path $specifyDir -Force | Out-Null
|
|
}
|
|
|
|
# Write feature.json
|
|
$json = @{ feature_directory = $FeatureDirectory } | ConvertTo-Json -Compress
|
|
$utf8NoBom = New-Object System.Text.UTF8Encoding($false)
|
|
[System.IO.File]::WriteAllText($fjPath, $json, $utf8NoBom)
|
|
}
|
|
|
|
function Get-FeaturePathsEnv {
|
|
$repoRoot = Get-RepoRoot
|
|
$currentBranch = Get-CurrentBranch
|
|
|
|
# Resolve feature directory. Priority:
|
|
# 1. SPECIFY_FEATURE_DIRECTORY env var (explicit override)
|
|
# 2. .specify/feature.json "feature_directory" key (persisted by specify command)
|
|
# 3. Error - no feature context available
|
|
$featureJson = Join-Path $repoRoot '.specify/feature.json'
|
|
if ($env:SPECIFY_FEATURE_DIRECTORY) {
|
|
$featureDir = $env:SPECIFY_FEATURE_DIRECTORY
|
|
# Normalize relative paths to absolute under repo root
|
|
if (-not [System.IO.Path]::IsPathRooted($featureDir)) {
|
|
$featureDir = Join-Path $repoRoot $featureDir
|
|
}
|
|
# Persist to feature.json so future sessions without the env var still work
|
|
Save-FeatureJson -RepoRoot $repoRoot -FeatureDirectory $env:SPECIFY_FEATURE_DIRECTORY
|
|
} elseif (Test-Path $featureJson) {
|
|
$featureJsonRaw = Get-Content -LiteralPath $featureJson -Raw
|
|
try {
|
|
$featureConfig = $featureJsonRaw | ConvertFrom-Json
|
|
} catch {
|
|
[Console]::Error.WriteLine("ERROR: Failed to parse .specify/feature.json: $_")
|
|
exit 1
|
|
}
|
|
if ($featureConfig.feature_directory) {
|
|
$featureDir = $featureConfig.feature_directory
|
|
# Normalize relative paths to absolute under repo root
|
|
if (-not [System.IO.Path]::IsPathRooted($featureDir)) {
|
|
$featureDir = Join-Path $repoRoot $featureDir
|
|
}
|
|
} else {
|
|
[Console]::Error.WriteLine("ERROR: Feature directory not found. Set SPECIFY_FEATURE_DIRECTORY or ensure .specify/feature.json contains feature_directory.")
|
|
exit 1
|
|
}
|
|
} else {
|
|
[Console]::Error.WriteLine("ERROR: Feature directory not found. Set SPECIFY_FEATURE_DIRECTORY or run the specify command to create .specify/feature.json.")
|
|
exit 1
|
|
}
|
|
|
|
[PSCustomObject]@{
|
|
REPO_ROOT = $repoRoot
|
|
CURRENT_BRANCH = $currentBranch
|
|
FEATURE_DIR = $featureDir
|
|
FEATURE_SPEC = Join-Path $featureDir 'spec.md'
|
|
IMPL_PLAN = Join-Path $featureDir 'plan.md'
|
|
TASKS = Join-Path $featureDir 'tasks.md'
|
|
RESEARCH = Join-Path $featureDir 'research.md'
|
|
DATA_MODEL = Join-Path $featureDir 'data-model.md'
|
|
QUICKSTART = Join-Path $featureDir 'quickstart.md'
|
|
CONTRACTS_DIR = Join-Path $featureDir 'contracts'
|
|
}
|
|
}
|
|
|
|
function Test-FileExists {
|
|
param([string]$Path, [string]$Description)
|
|
if (Test-Path -Path $Path -PathType Leaf) {
|
|
Write-Output " [OK] $Description"
|
|
return $true
|
|
} else {
|
|
Write-Output " [FAIL] $Description"
|
|
return $false
|
|
}
|
|
}
|
|
|
|
function Test-DirHasFiles {
|
|
param([string]$Path, [string]$Description)
|
|
# A directory counts as non-empty when Get-ChildItem returns any entry
|
|
# (files or subdirectories) -- matching the JSON contracts checks in
|
|
# check-prerequisites.ps1 / setup-tasks.ps1, and treating a directory whose
|
|
# only contents are subdirectories (e.g. contracts/v1/openapi.yaml) as
|
|
# non-empty like bash check_dir. Filtering out subdirectories would
|
|
# mis-report such a directory as empty.
|
|
if ((Test-Path -Path $Path -PathType Container) -and (Get-ChildItem -Path $Path -ErrorAction SilentlyContinue | Select-Object -First 1)) {
|
|
Write-Output " [OK] $Description"
|
|
return $true
|
|
} else {
|
|
Write-Output " [FAIL] $Description"
|
|
return $false
|
|
}
|
|
}
|
|
|
|
function Get-InvokeSeparator {
|
|
param([string]$RepoRoot = (Get-RepoRoot))
|
|
|
|
if ($null -eq $script:SpecKitInvokeSeparatorCache) {
|
|
$script:SpecKitInvokeSeparatorCache = @{}
|
|
}
|
|
if ($script:SpecKitInvokeSeparatorCache.ContainsKey($RepoRoot)) {
|
|
return $script:SpecKitInvokeSeparatorCache[$RepoRoot]
|
|
}
|
|
|
|
$separator = '.'
|
|
$integrationJson = Join-Path $RepoRoot '.specify/integration.json'
|
|
if (Test-Path -LiteralPath $integrationJson -PathType Leaf) {
|
|
try {
|
|
$state = Get-Content -LiteralPath $integrationJson -Raw | ConvertFrom-Json
|
|
$key = if ($state.default_integration) { [string]$state.default_integration } elseif ($state.integration) { [string]$state.integration } else { '' }
|
|
if ($key -and $state.integration_settings) {
|
|
$settingProperty = $state.integration_settings.PSObject.Properties[$key]
|
|
if ($settingProperty) {
|
|
$setting = $settingProperty.Value
|
|
if ($setting -and ($setting.invoke_separator -eq '.' -or $setting.invoke_separator -eq '-')) {
|
|
$separator = [string]$setting.invoke_separator
|
|
}
|
|
}
|
|
}
|
|
} catch {
|
|
$separator = '.'
|
|
}
|
|
}
|
|
|
|
$script:SpecKitInvokeSeparatorCache[$RepoRoot] = $separator
|
|
return $separator
|
|
}
|
|
|
|
function Format-SpecKitCommand {
|
|
param(
|
|
[Parameter(Mandatory = $true)][string]$CommandName,
|
|
[string]$RepoRoot = (Get-RepoRoot)
|
|
)
|
|
|
|
$separator = Get-InvokeSeparator -RepoRoot $RepoRoot
|
|
$name = $CommandName.TrimStart('/')
|
|
if ($name.StartsWith('speckit.')) {
|
|
$name = $name.Substring(8)
|
|
} elseif ($name.StartsWith('speckit-')) {
|
|
$name = $name.Substring(8)
|
|
}
|
|
$name = $name -replace '\.', $separator
|
|
|
|
return "/speckit$separator$name"
|
|
}
|
|
|
|
# Find a usable Python 3 executable (python3, python, or py -3).
|
|
# Returns the command/arguments as an array, or $null if none found.
|
|
function Get-Python3Command {
|
|
if (Get-Command python3 -ErrorAction SilentlyContinue) { return @('python3') }
|
|
if (Get-Command python -ErrorAction SilentlyContinue) {
|
|
$ver = & python --version 2>&1
|
|
if ($ver -match 'Python 3') { return @('python') }
|
|
}
|
|
if (Get-Command py -ErrorAction SilentlyContinue) {
|
|
$ver = & py -3 --version 2>&1
|
|
if ($ver -match 'Python 3') { return @('py', '-3') }
|
|
}
|
|
return $null
|
|
}
|
|
|
|
# Resolve a template name to a file path using the priority stack:
|
|
# 1. .specify/templates/overrides/
|
|
# 2. .specify/presets/<preset-id>/templates/ (sorted by priority from .registry)
|
|
# 3. .specify/extensions/<ext-id>/templates/
|
|
# 4. .specify/templates/ (core)
|
|
function Resolve-Template {
|
|
param(
|
|
[Parameter(Mandatory=$true)][string]$TemplateName,
|
|
[Parameter(Mandatory=$true)][string]$RepoRoot
|
|
)
|
|
|
|
$base = Join-Path $RepoRoot '.specify/templates'
|
|
|
|
# Priority 1: Project overrides
|
|
$override = Join-Path $base "overrides/$TemplateName.md"
|
|
if (Test-Path $override) { return $override }
|
|
|
|
# Priority 2: Installed presets (sorted by priority from .registry)
|
|
$presetsDir = Join-Path $RepoRoot '.specify/presets'
|
|
if (Test-Path $presetsDir) {
|
|
$registryFile = Join-Path $presetsDir '.registry'
|
|
$sortedPresets = @()
|
|
if (Test-Path $registryFile) {
|
|
try {
|
|
$registryData = Get-Content $registryFile -Raw | ConvertFrom-Json
|
|
$presets = $registryData.presets
|
|
if ($presets) {
|
|
$sortedPresets = $presets.PSObject.Properties |
|
|
Where-Object { $null -eq $_.Value.enabled -or $_.Value.enabled -ne $false } |
|
|
Sort-Object { if ($null -ne $_.Value.priority) { $_.Value.priority } else { 10 } } |
|
|
ForEach-Object { $_.Name }
|
|
}
|
|
} catch {
|
|
# Fallback: alphabetical directory order
|
|
$sortedPresets = @()
|
|
}
|
|
}
|
|
|
|
if ($sortedPresets.Count -gt 0) {
|
|
foreach ($presetId in $sortedPresets) {
|
|
$candidate = Join-Path $presetsDir "$presetId/templates/$TemplateName.md"
|
|
if (Test-Path $candidate) { return $candidate }
|
|
}
|
|
} else {
|
|
# Fallback: alphabetical directory order
|
|
foreach ($preset in Get-ChildItem -Path $presetsDir -Directory -ErrorAction SilentlyContinue | Where-Object { $_.Name -notlike '.*' }) {
|
|
$candidate = Join-Path $preset.FullName "templates/$TemplateName.md"
|
|
if (Test-Path $candidate) { return $candidate }
|
|
}
|
|
}
|
|
}
|
|
|
|
# Priority 3: Extension-provided templates
|
|
$extDir = Join-Path $RepoRoot '.specify/extensions'
|
|
if (Test-Path $extDir) {
|
|
foreach ($ext in Get-ChildItem -Path $extDir -Directory -ErrorAction SilentlyContinue | Where-Object { $_.Name -notlike '.*' } | Sort-Object Name) {
|
|
$candidate = Join-Path $ext.FullName "templates/$TemplateName.md"
|
|
if (Test-Path $candidate) { return $candidate }
|
|
}
|
|
}
|
|
|
|
# Priority 4: Core templates
|
|
$core = Join-Path $base "$TemplateName.md"
|
|
if (Test-Path $core) { return $core }
|
|
|
|
return $null
|
|
}
|
|
|
|
# Resolve a template name to composed content using composition strategies.
|
|
# Reads strategy metadata from preset manifests and composes content
|
|
# from multiple layers using prepend, append, or wrap strategies.
|
|
function Resolve-TemplateContent {
|
|
param(
|
|
[Parameter(Mandatory=$true)][string]$TemplateName,
|
|
[Parameter(Mandatory=$true)][string]$RepoRoot
|
|
)
|
|
|
|
$base = Join-Path $RepoRoot '.specify/templates'
|
|
|
|
# Collect all layers (highest priority first)
|
|
$layerPaths = @()
|
|
$layerStrategies = @()
|
|
|
|
# Priority 1: Project overrides (always "replace")
|
|
$override = Join-Path $base "overrides/$TemplateName.md"
|
|
if (Test-Path $override) {
|
|
$layerPaths += $override
|
|
$layerStrategies += 'replace'
|
|
}
|
|
|
|
# Priority 2: Installed presets (sorted by priority from .registry)
|
|
$presetsDir = Join-Path $RepoRoot '.specify/presets'
|
|
if (Test-Path $presetsDir) {
|
|
$registryFile = Join-Path $presetsDir '.registry'
|
|
$sortedPresets = @()
|
|
if (Test-Path $registryFile) {
|
|
try {
|
|
$registryData = Get-Content $registryFile -Raw | ConvertFrom-Json
|
|
$presets = $registryData.presets
|
|
if ($presets) {
|
|
$sortedPresets = $presets.PSObject.Properties |
|
|
Where-Object { $null -eq $_.Value.enabled -or $_.Value.enabled -ne $false } |
|
|
Sort-Object { if ($null -ne $_.Value.priority) { $_.Value.priority } else { 10 } } |
|
|
ForEach-Object { $_.Name }
|
|
}
|
|
} catch {
|
|
$sortedPresets = @()
|
|
}
|
|
}
|
|
|
|
if ($sortedPresets.Count -gt 0) {
|
|
$pyCmd = Get-Python3Command
|
|
if (-not $pyCmd) {
|
|
# Check if any preset has strategy fields that would be ignored
|
|
foreach ($pid in $sortedPresets) {
|
|
$mf = Join-Path $presetsDir "$pid/preset.yml"
|
|
if ((Test-Path $mf) -and (Select-String -Path $mf -Pattern 'strategy:' -Quiet -ErrorAction SilentlyContinue)) {
|
|
Write-Warning "No Python 3 found; preset composition strategies will be ignored"
|
|
break
|
|
}
|
|
}
|
|
}
|
|
$yamlWarned = $false
|
|
foreach ($presetId in $sortedPresets) {
|
|
# Read strategy and file path from preset manifest
|
|
$strategy = 'replace'
|
|
$manifestFilePath = ''
|
|
$manifest = Join-Path $presetsDir "$presetId/preset.yml"
|
|
if ((Test-Path $manifest) -and $pyCmd) {
|
|
try {
|
|
# Use Python to parse YAML manifest for strategy and file path
|
|
$pyArgs = if ($pyCmd.Count -gt 1) { $pyCmd[1..($pyCmd.Count-1)] } else { @() }
|
|
$pyStderrFile = [System.IO.Path]::GetTempFileName()
|
|
$stratResult = & $pyCmd[0] @pyArgs -c @"
|
|
import sys
|
|
try:
|
|
import yaml
|
|
except ImportError:
|
|
print('yaml_missing', file=sys.stderr)
|
|
print('replace\t')
|
|
sys.exit(0)
|
|
try:
|
|
with open(sys.argv[1]) as f:
|
|
data = yaml.safe_load(f)
|
|
for t in data.get('provides', {}).get('templates', []):
|
|
if t.get('name') == sys.argv[2] and t.get('type', 'template') == 'template':
|
|
print(t.get('strategy', 'replace') + '\t' + t.get('file', ''))
|
|
sys.exit(0)
|
|
print('replace\t')
|
|
except Exception:
|
|
print('replace\t')
|
|
"@ $manifest $TemplateName 2>$pyStderrFile
|
|
if ($stratResult) {
|
|
$parts = $stratResult.Trim() -split "`t", 2
|
|
$strategy = $parts[0].ToLowerInvariant()
|
|
if ($parts.Count -gt 1 -and $parts[1]) { $manifestFilePath = $parts[1] }
|
|
}
|
|
if (-not $yamlWarned -and (Test-Path $pyStderrFile) -and (Get-Content $pyStderrFile -Raw -ErrorAction SilentlyContinue) -match 'yaml_missing') {
|
|
Write-Warning "PyYAML not available; composition strategies may be ignored"
|
|
$yamlWarned = $true
|
|
}
|
|
Remove-Item $pyStderrFile -Force -ErrorAction SilentlyContinue
|
|
} catch {
|
|
$strategy = 'replace'
|
|
if ($pyStderrFile) { Remove-Item $pyStderrFile -Force -ErrorAction SilentlyContinue }
|
|
}
|
|
}
|
|
# Try manifest file path first, then convention path
|
|
$candidate = $null
|
|
if ($manifestFilePath) {
|
|
# Reject absolute paths and parent traversal
|
|
if ([System.IO.Path]::IsPathRooted($manifestFilePath) -or $manifestFilePath -match '\.\.[\\/]') {
|
|
$manifestFilePath = ''
|
|
}
|
|
}
|
|
if ($manifestFilePath) {
|
|
$mf = Join-Path $presetsDir "$presetId/$manifestFilePath"
|
|
if (Test-Path $mf) { $candidate = $mf }
|
|
}
|
|
if (-not $candidate) {
|
|
$cf = Join-Path $presetsDir "$presetId/templates/$TemplateName.md"
|
|
if (Test-Path $cf) { $candidate = $cf }
|
|
}
|
|
if ($candidate) {
|
|
$layerPaths += $candidate
|
|
$layerStrategies += $strategy
|
|
}
|
|
}
|
|
} else {
|
|
# Fallback: alphabetical directory order (no registry or parse failure)
|
|
foreach ($preset in Get-ChildItem -Path $presetsDir -Directory -ErrorAction SilentlyContinue | Where-Object { $_.Name -notlike '.*' }) {
|
|
$candidate = Join-Path $preset.FullName "templates/$TemplateName.md"
|
|
if (Test-Path $candidate) {
|
|
$layerPaths += $candidate
|
|
$layerStrategies += 'replace'
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# Priority 3: Extension-provided templates (always "replace")
|
|
$extDir = Join-Path $RepoRoot '.specify/extensions'
|
|
if (Test-Path $extDir) {
|
|
foreach ($ext in Get-ChildItem -Path $extDir -Directory -ErrorAction SilentlyContinue | Where-Object { $_.Name -notlike '.*' } | Sort-Object Name) {
|
|
$candidate = Join-Path $ext.FullName "templates/$TemplateName.md"
|
|
if (Test-Path $candidate) {
|
|
$layerPaths += $candidate
|
|
$layerStrategies += 'replace'
|
|
}
|
|
}
|
|
}
|
|
|
|
# Priority 4: Core templates (always "replace")
|
|
$core = Join-Path $base "$TemplateName.md"
|
|
if (Test-Path $core) {
|
|
$layerPaths += $core
|
|
$layerStrategies += 'replace'
|
|
}
|
|
|
|
if ($layerPaths.Count -eq 0) { return $null }
|
|
|
|
# If the top (highest-priority) layer is replace, it wins entirely --
|
|
# lower layers are irrelevant regardless of their strategies.
|
|
if ($layerStrategies[0] -eq 'replace') {
|
|
return (Get-Content $layerPaths[0] -Raw)
|
|
}
|
|
|
|
# Check if any layer uses a non-replace strategy
|
|
$hasComposition = $false
|
|
foreach ($s in $layerStrategies) {
|
|
if ($s -ne 'replace') { $hasComposition = $true; break }
|
|
}
|
|
|
|
if (-not $hasComposition) {
|
|
return (Get-Content $layerPaths[0] -Raw)
|
|
}
|
|
|
|
# Find the effective base: scan from highest priority (index 0) downward
|
|
# to find the nearest replace layer. Only compose layers above that base.
|
|
$baseIdx = -1
|
|
for ($i = 0; $i -lt $layerPaths.Count; $i++) {
|
|
if ($layerStrategies[$i] -eq 'replace') {
|
|
$baseIdx = $i
|
|
break
|
|
}
|
|
}
|
|
if ($baseIdx -lt 0) { return $null }
|
|
|
|
$content = Get-Content $layerPaths[$baseIdx] -Raw
|
|
|
|
for ($i = $baseIdx - 1; $i -ge 0; $i--) {
|
|
$path = $layerPaths[$i]
|
|
$strat = $layerStrategies[$i]
|
|
$layerContent = Get-Content $path -Raw
|
|
|
|
switch ($strat) {
|
|
'replace' { $content = $layerContent }
|
|
'prepend' { $content = "$layerContent`n`n$content" }
|
|
'append' { $content = "$content`n`n$layerContent" }
|
|
'wrap' {
|
|
if (-not $layerContent.Contains('{CORE_TEMPLATE}')) {
|
|
throw "Wrap strategy missing {CORE_TEMPLATE} placeholder"
|
|
}
|
|
$content = $layerContent.Replace('{CORE_TEMPLATE}', $content)
|
|
}
|
|
default { throw "Unknown strategy: $strat" }
|
|
}
|
|
}
|
|
|
|
return $content
|
|
}
|