# subsystem-compile v1.0 — Create 1C subsystem from JSON definition # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills param( [string]$DefinitionFile, [string]$Value, [Parameter(Mandatory)][string]$OutputDir, [string]$Parent, [switch]$NoValidate ) $ErrorActionPreference = "Stop" [Console]::OutputEncoding = [System.Text.Encoding]::UTF8 # --- 1. Load JSON --- if ($DefinitionFile -and $Value) { Write-Error "Cannot use both -DefinitionFile and -Value" exit 1 } if (-not $DefinitionFile -and -not $Value) { Write-Error "Either -DefinitionFile or -Value is required" exit 1 } if ($DefinitionFile) { if (-not [System.IO.Path]::IsPathRooted($DefinitionFile)) { $DefinitionFile = Join-Path (Get-Location).Path $DefinitionFile } if (-not (Test-Path $DefinitionFile)) { Write-Error "Definition file not found: $DefinitionFile" exit 1 } $json = Get-Content -Raw -Encoding UTF8 $DefinitionFile } else { $json = $Value } $def = $json | ConvertFrom-Json if (-not $def.name) { Write-Error "JSON must have 'name' field" exit 1 } $objName = "$($def.name)" # Resolve OutputDir if (-not [System.IO.Path]::IsPathRooted($OutputDir)) { $OutputDir = Join-Path (Get-Location).Path $OutputDir } # --- 2. XML helpers --- $script:xml = New-Object System.Text.StringBuilder 8192 function X([string]$text) { $script:xml.AppendLine($text) | Out-Null } function Esc-Xml([string]$s) { return $s.Replace('&','&').Replace('<','<').Replace('>','>').Replace('"','"') } function Split-CamelCase([string]$name) { if (-not $name) { return $name } $result = [regex]::Replace($name, '([a-z\u0430-\u044F\u0451])([A-Z\u0410-\u042F\u0401])', '$1 $2') if ($result.Length -gt 1) { $result = $result.Substring(0,1) + $result.Substring(1).ToLower() } return $result } function Emit-MLText([string]$indent, [string]$tag, [string]$text) { if (-not $text) { X "$indent<$tag/>" return } X "$indent<$tag>" X "$indent`t" X "$indent`t`tru" X "$indent`t`t$(Esc-Xml $text)" X "$indent`t" X "$indent" } function New-Guid-String { return [System.Guid]::NewGuid().ToString() } # --- 3. Resolve defaults --- $synonym = if ($def.synonym) { "$($def.synonym)" } else { Split-CamelCase $objName } $comment = if ($def.comment) { "$($def.comment)" } else { "" } $includeHelpInContents = "true" $includeInCI = if ($null -ne $def.includeInCommandInterface) { "$($def.includeInCommandInterface)".ToLower() } else { "true" } $useOneCommand = if ($null -ne $def.useOneCommand) { "$($def.useOneCommand)".ToLower() } else { "false" } $explanation = if ($def.explanation) { "$($def.explanation)" } else { "" } $picture = if ($def.picture) { "$($def.picture)" } else { "" } $contentItems = @() if ($def.content) { foreach ($c in $def.content) { $contentItems += "$c" } } $children = @() if ($def.children) { foreach ($ch in $def.children) { $children += "$ch" } } # --- 4. Build XML --- $uuid = New-Guid-String $indent = "`t`t`t" X '' X '' X "`t" X "`t`t" # Name X "`t`t`t$(Esc-Xml $objName)" # Synonym Emit-MLText "`t`t`t" "Synonym" $synonym # Comment if ($comment) { X "`t`t`t$(Esc-Xml $comment)" } else { X "`t`t`t" } # Boolean properties X "`t`t`t$includeHelpInContents" X "`t`t`t$includeInCI" X "`t`t`t$useOneCommand" # Explanation Emit-MLText "`t`t`t" "Explanation" $explanation # Picture if ($picture) { X "`t`t`t" X "`t`t`t`t$picture" X "`t`t`t`tfalse" X "`t`t`t" } else { X "`t`t`t" } # Content if ($contentItems.Count -gt 0) { X "`t`t`t" foreach ($item in $contentItems) { X "`t`t`t`t$(Esc-Xml $item)" } X "`t`t`t" } else { X "`t`t`t" } X "`t`t" # ChildObjects if ($children.Count -gt 0) { X "`t`t" foreach ($ch in $children) { X "`t`t`t$(Esc-Xml $ch)" } X "`t`t" } else { X "`t`t" } X "`t" X '' # --- 5. Write files --- # Determine target directory if ($Parent) { # Nested subsystem if (-not [System.IO.Path]::IsPathRooted($Parent)) { $Parent = Join-Path (Get-Location).Path $Parent } if (-not (Test-Path $Parent)) { Write-Error "Parent subsystem not found: $Parent" exit 1 } $parentDir = [System.IO.Path]::GetDirectoryName($Parent) $parentBaseName = [System.IO.Path]::GetFileNameWithoutExtension($Parent) $subsDir = Join-Path (Join-Path $parentDir $parentBaseName) "Subsystems" } else { # Top-level subsystem $subsDir = Join-Path $OutputDir "Subsystems" } if (-not (Test-Path $subsDir)) { New-Item -ItemType Directory -Path $subsDir -Force | Out-Null } $targetXml = Join-Path $subsDir "$objName.xml" # Write XML $xmlContent = $script:xml.ToString() $utf8Bom = New-Object System.Text.UTF8Encoding($true) [System.IO.File]::WriteAllText($targetXml, $xmlContent, $utf8Bom) Write-Host "[OK] Created: $targetXml" # Create subdirectory if children exist if ($children.Count -gt 0) { $childSubsDir = Join-Path (Join-Path $subsDir $objName) "Subsystems" if (-not (Test-Path $childSubsDir)) { New-Item -ItemType Directory -Path $childSubsDir -Force | Out-Null Write-Host "[OK] Created directory: $childSubsDir" } } # --- 6. Register in parent --- $parentXmlPath = $null if ($Parent) { $parentXmlPath = $Parent } else { $configXml = Join-Path $OutputDir "Configuration.xml" if (Test-Path $configXml) { $parentXmlPath = $configXml } } if ($parentXmlPath -and (Test-Path $parentXmlPath)) { $doc = New-Object System.Xml.XmlDocument $doc.PreserveWhitespace = $true $doc.Load($parentXmlPath) $ns = New-Object System.Xml.XmlNamespaceManager($doc.NameTable) $ns.AddNamespace("md", "http://v8.1c.ru/8.3/MDClasses") # Find ChildObjects $childObjects = $null if ($Parent) { $childObjects = $doc.SelectSingleNode("//md:Subsystem/md:ChildObjects", $ns) } else { $childObjects = $doc.SelectSingleNode("//md:Configuration/md:ChildObjects", $ns) } if ($childObjects) { # Check for self-closing tag $isSelfClosing = (-not $childObjects.HasChildNodes) -or ($childObjects.IsEmpty) # Check if already registered $alreadyExists = $false foreach ($child in $childObjects.ChildNodes) { if ($child.NodeType -eq 'Element' -and $child.LocalName -eq "Subsystem" -and $child.InnerText -eq $objName) { $alreadyExists = $true break } } if (-not $alreadyExists) { $newEl = $doc.CreateElement("Subsystem", "http://v8.1c.ru/8.3/MDClasses") $newEl.InnerText = $objName if ($isSelfClosing) { # Expand self-closing tag $parentIndent = "" $prev = $childObjects.PreviousSibling if ($prev -and ($prev.NodeType -eq 'Whitespace' -or $prev.NodeType -eq 'SignificantWhitespace')) { if ($prev.Value -match '(\t+)$') { $parentIndent = $Matches[1] } } $childIndent = "$parentIndent`t" $ws1 = $doc.CreateWhitespace("`r`n$childIndent") $ws2 = $doc.CreateWhitespace("`r`n$parentIndent") $childObjects.AppendChild($ws1) | Out-Null $childObjects.AppendChild($newEl) | Out-Null $childObjects.AppendChild($ws2) | Out-Null } else { # Insert before trailing whitespace $childIndent = "`t`t`t" foreach ($child in $childObjects.ChildNodes) { if ($child.NodeType -eq 'Whitespace' -or $child.NodeType -eq 'SignificantWhitespace') { if ($child.Value -match '^\r?\n(\t+)') { $childIndent = $Matches[1]; break } } } $trailing = $childObjects.LastChild $ws = $doc.CreateWhitespace("`r`n$childIndent") if ($trailing -and ($trailing.NodeType -eq 'Whitespace' -or $trailing.NodeType -eq 'SignificantWhitespace')) { $childObjects.InsertBefore($ws, $trailing) | Out-Null $childObjects.InsertBefore($newEl, $trailing) | Out-Null } else { $childObjects.AppendChild($ws) | Out-Null $childObjects.AppendChild($newEl) | Out-Null } } # Save parent XML $settings = New-Object System.Xml.XmlWriterSettings $settings.Encoding = New-Object System.Text.UTF8Encoding($true) $settings.Indent = $false $settings.NewLineHandling = [System.Xml.NewLineHandling]::None $memStream = New-Object System.IO.MemoryStream $writer = [System.Xml.XmlWriter]::Create($memStream, $settings) $doc.Save($writer) $writer.Flush(); $writer.Close() $bytes = $memStream.ToArray() $memStream.Close() $text = [System.Text.Encoding]::UTF8.GetString($bytes) if ($text.Length -gt 0 -and $text[0] -eq [char]0xFEFF) { $text = $text.Substring(1) } $text = $text.Replace('encoding="utf-8"', 'encoding="UTF-8"') [System.IO.File]::WriteAllText($parentXmlPath, $text, $utf8Bom) Write-Host "[OK] Registered in: $parentXmlPath" } else { Write-Host "[SKIP] Already registered in: $parentXmlPath" } } else { Write-Host "[WARN] ChildObjects not found in: $parentXmlPath" } } else { Write-Host "[INFO] No parent XML to register in" } # --- 7. Auto-validate --- if (-not $NoValidate) { $validateScript = Join-Path (Join-Path $PSScriptRoot "..\..\subsystem-validate") "scripts\subsystem-validate.ps1" $validateScript = [System.IO.Path]::GetFullPath($validateScript) if (Test-Path $validateScript) { Write-Host "" Write-Host "--- Running subsystem-validate ---" & powershell.exe -NoProfile -File $validateScript -SubsystemPath $targetXml } } Write-Host "" Write-Host "=== subsystem-compile summary ===" Write-Host " Name: $objName" Write-Host " UUID: $uuid" Write-Host " Content: $($contentItems.Count) objects" Write-Host " Children: $($children.Count)" Write-Host " File: $targetXml" exit 0