# meta-remove v1.1 — Remove metadata object from 1C configuration dump # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills param( [Parameter(Mandatory)] [string]$ConfigDir, [Parameter(Mandatory)] [string]$Object, [switch]$DryRun, [switch]$KeepFiles, [switch]$Force ) $ErrorActionPreference = "Stop" [Console]::OutputEncoding = [System.Text.Encoding]::UTF8 # --- Type → plural directory mapping --- $typePluralMap = @{ "Catalog" = "Catalogs" "Document" = "Documents" "Enum" = "Enums" "Constant" = "Constants" "InformationRegister" = "InformationRegisters" "AccumulationRegister" = "AccumulationRegisters" "AccountingRegister" = "AccountingRegisters" "CalculationRegister" = "CalculationRegisters" "ChartOfAccounts" = "ChartsOfAccounts" "ChartOfCharacteristicTypes" = "ChartsOfCharacteristicTypes" "ChartOfCalculationTypes" = "ChartsOfCalculationTypes" "BusinessProcess" = "BusinessProcesses" "Task" = "Tasks" "ExchangePlan" = "ExchangePlans" "DocumentJournal" = "DocumentJournals" "Report" = "Reports" "DataProcessor" = "DataProcessors" "CommonModule" = "CommonModules" "ScheduledJob" = "ScheduledJobs" "EventSubscription" = "EventSubscriptions" "HTTPService" = "HTTPServices" "WebService" = "WebServices" "DefinedType" = "DefinedTypes" "Role" = "Roles" "Subsystem" = "Subsystems" "CommonForm" = "CommonForms" "CommonTemplate" = "CommonTemplates" "CommonPicture" = "CommonPictures" "CommonAttribute" = "CommonAttributes" "SessionParameter" = "SessionParameters" "FunctionalOption" = "FunctionalOptions" "FunctionalOptionsParameter" = "FunctionalOptionsParameters" "Sequence" = "Sequences" "FilterCriterion" = "FilterCriteria" "SettingsStorage" = "SettingsStorages" "XDTOPackage" = "XDTOPackages" "WSReference" = "WSReferences" "StyleItem" = "StyleItems" "Language" = "Languages" } # --- Resolve paths --- if (-not [System.IO.Path]::IsPathRooted($ConfigDir)) { $ConfigDir = Join-Path (Get-Location).Path $ConfigDir } if (-not (Test-Path $ConfigDir -PathType Container)) { Write-Host "[ERROR] Config directory not found: $ConfigDir" exit 1 } $configXml = Join-Path $ConfigDir "Configuration.xml" if (-not (Test-Path $configXml)) { Write-Host "[ERROR] Configuration.xml not found in: $ConfigDir" exit 1 } # --- Parse object spec --- $parts = $Object -split "\.", 2 if ($parts.Count -ne 2 -or -not $parts[0] -or -not $parts[1]) { Write-Host "[ERROR] Invalid object format '$Object'. Expected: Type.Name (e.g. Catalog.Товары)" exit 1 } $objType = $parts[0] $objName = $parts[1] if (-not $typePluralMap.ContainsKey($objType)) { Write-Host "[ERROR] Unknown type '$objType'. Supported: $($typePluralMap.Keys -join ', ')" exit 1 } $typePlural = $typePluralMap[$objType] Write-Host "=== meta-remove: ${objType}.${objName} ===" Write-Host "" if ($DryRun) { Write-Host "[DRY-RUN] No changes will be made" Write-Host "" } $actions = 0 $errors = 0 # --- 1. Find object files --- $typeDir = Join-Path $ConfigDir $typePlural $objXml = Join-Path $typeDir "$objName.xml" $objDir = Join-Path $typeDir $objName $hasXml = Test-Path $objXml $hasDir = Test-Path $objDir -PathType Container if (-not $hasXml -and -not $hasDir) { # Check if registered in Configuration.xml before proceeding $cfgCheckDoc = New-Object System.Xml.XmlDocument $cfgCheckDoc.PreserveWhitespace = $true $cfgCheckDoc.Load($configXml) $cfgCheckNs = New-Object System.Xml.XmlNamespaceManager($cfgCheckDoc.NameTable) $cfgCheckNs.AddNamespace("md", "http://v8.1c.ru/8.3/MDClasses") $cfgCheckNode = $cfgCheckDoc.DocumentElement.SelectSingleNode("md:Configuration/md:ChildObjects", $cfgCheckNs) $registeredInCfg = $false if ($cfgCheckNode) { foreach ($child in @($cfgCheckNode.ChildNodes)) { if ($child.NodeType -ne 'Element') { continue } if ($child.LocalName -eq $objType -and $child.InnerText.Trim() -eq $objName) { $registeredInCfg = $true; break } } } if (-not $registeredInCfg) { Write-Host "[ERROR] Object not found: $typePlural/$objName.xml and not registered in Configuration.xml" exit 1 } Write-Host "[WARN] Object files not found: $typePlural/$objName.xml" Write-Host " Proceeding with deregistration only..." } else { if ($hasXml) { Write-Host "[FOUND] $typePlural/$objName.xml" } if ($hasDir) { $fileCount = @(Get-ChildItem $objDir -Recurse -File).Count Write-Host "[FOUND] $typePlural/$objName/ ($fileCount files)" } } # --- 2. Reference check --- Write-Host "" Write-Host "--- Reference check ---" # Build search patterns based on object type # Type → reference type name (used in XML elements) $typeRefNames = @{ "Catalog" = @("CatalogRef","CatalogObject") "Document" = @("DocumentRef","DocumentObject") "Enum" = @("EnumRef") "ExchangePlan" = @("ExchangePlanRef","ExchangePlanObject") "ChartOfAccounts" = @("ChartOfAccountsRef","ChartOfAccountsObject") "ChartOfCharacteristicTypes" = @("ChartOfCharacteristicTypesRef","ChartOfCharacteristicTypesObject") "ChartOfCalculationTypes" = @("ChartOfCalculationTypesRef","ChartOfCalculationTypesObject") "BusinessProcess" = @("BusinessProcessRef","BusinessProcessObject") "Task" = @("TaskRef","TaskObject") } # Type → Russian manager name (used in BSL code: Справочники.Товары) $typeRuManager = @{ "Catalog" = "Справочники" "Document" = "Документы" "Enum" = "Перечисления" "Constant" = "Константы" "InformationRegister" = "РегистрыСведений" "AccumulationRegister" = "РегистрыНакопления" "AccountingRegister" = "РегистрыБухгалтерии" "CalculationRegister" = "РегистрыРасчета" "ChartOfAccounts" = "ПланыСчетов" "ChartOfCharacteristicTypes" = "ПланыВидовХарактеристик" "ChartOfCalculationTypes" = "ПланыВидовРасчета" "BusinessProcess" = "БизнесПроцессы" "Task" = "Задачи" "ExchangePlan" = "ПланыОбмена" "Report" = "Отчеты" "DataProcessor" = "Обработки" "DocumentJournal" = "ЖурналыДокументов" "CommonModule" = $null } $searchPatterns = @() # 1) XML type references: CatalogRef.Name, CatalogObject.Name if ($typeRefNames.ContainsKey($objType)) { foreach ($refName in $typeRefNames[$objType]) { $searchPatterns += "$refName.$objName" } } # 2) BSL code references: Справочники.Name, Catalogs.Name $ruMgr = $typeRuManager[$objType] if ($ruMgr) { $searchPatterns += "$ruMgr.$objName" } # English manager = plural directory name $searchPatterns += "$typePlural.$objName" # 3) CommonModule: method calls in BSL (ModuleName.) if ($objType -eq "CommonModule") { $searchPatterns += "$objName." } # 4) ScheduledJob/EventSubscription handler references if ($objType -eq "CommonModule") { $searchPatterns += "$objName." $searchPatterns += "$objName." } # Exclude object's own files from search $excludeDirs = @() if ($hasDir) { $excludeDirs += $objDir } $excludeFile = "" if ($hasXml) { $excludeFile = $objXml } # Search all XML and BSL files $references = @() $searchExtensions = @("*.xml", "*.bsl") foreach ($ext in $searchExtensions) { $files = @(Get-ChildItem $ConfigDir -Filter $ext -Recurse -File -ErrorAction SilentlyContinue) foreach ($file in $files) { # Skip own files if ($excludeFile -and $file.FullName -eq $excludeFile) { continue } if ($excludeDirs.Count -gt 0) { $skip = $false foreach ($ed in $excludeDirs) { if ($file.FullName.StartsWith($ed)) { $skip = $true; break } } if ($skip) { continue } } # Skip auto-cleaned files (Configuration.xml, ConfigDumpInfo.xml, Subsystems) $relPath = $file.FullName.Substring($ConfigDir.Length + 1) if ($relPath -eq "Configuration.xml" -or $relPath -eq "ConfigDumpInfo.xml" -or $relPath.StartsWith("Subsystems")) { continue } $content = [System.IO.File]::ReadAllText($file.FullName, [System.Text.Encoding]::UTF8) foreach ($pat in $searchPatterns) { if ($content.Contains($pat)) { $references += @{ File = $relPath; Pattern = $pat } break # one match per file is enough } } } } # Also check for Type.Name references (subsystem content, doc journal, etc.) — but NOT in own files $typeNameRef = "${objType}.${objName}" $files = @(Get-ChildItem $ConfigDir -Filter "*.xml" -Recurse -File -ErrorAction SilentlyContinue) foreach ($file in $files) { if ($excludeFile -and $file.FullName -eq $excludeFile) { continue } if ($excludeDirs.Count -gt 0) { $skip = $false foreach ($ed in $excludeDirs) { if ($file.FullName.StartsWith($ed)) { $skip = $true; break } } if ($skip) { continue } } # Skip Configuration.xml and Subsystems — they will be cleaned automatically $relPath = $file.FullName.Substring($ConfigDir.Length + 1) if ($relPath -eq "Configuration.xml") { continue } if ($relPath -eq "ConfigDumpInfo.xml") { continue } if ($relPath.StartsWith("Subsystems")) { continue } $content = [System.IO.File]::ReadAllText($file.FullName, [System.Text.Encoding]::UTF8) if ($content.Contains($typeNameRef)) { # Check it's not already in references $alreadyFound = $false foreach ($r in $references) { if ($r.File -eq $relPath) { $alreadyFound = $true; break } } if (-not $alreadyFound) { $references += @{ File = $relPath; Pattern = $typeNameRef } } } } if ($references.Count -gt 0) { Write-Host "[WARN] Found $($references.Count) reference(s) to ${objType}.${objName}:" Write-Host "" $shown = 0 foreach ($ref in $references) { Write-Host " $($ref.File)" Write-Host " pattern: $($ref.Pattern)" $shown++ if ($shown -ge 20) { $remaining = $references.Count - $shown if ($remaining -gt 0) { Write-Host " ... and $remaining more" } break } } Write-Host "" if (-not $Force) { Write-Host "[ERROR] Cannot remove: object has $($references.Count) reference(s)." Write-Host " Use -Force to remove anyway, or fix references first." exit 1 } else { Write-Host "[WARN] -Force specified, proceeding despite references" } } else { Write-Host "[OK] No references found" } # --- 3. Remove from Configuration.xml ChildObjects --- Write-Host "" Write-Host "--- Configuration.xml ---" $xmlDoc = New-Object System.Xml.XmlDocument $xmlDoc.PreserveWhitespace = $true $xmlDoc.Load($configXml) $ns = New-Object System.Xml.XmlNamespaceManager($xmlDoc.NameTable) $ns.AddNamespace("md", "http://v8.1c.ru/8.3/MDClasses") $ns.AddNamespace("v8", "http://v8.1c.ru/8.1/data/core") $cfgNode = $xmlDoc.DocumentElement.SelectSingleNode("md:Configuration", $ns) if (-not $cfgNode) { Write-Host "[ERROR] Configuration element not found in Configuration.xml" $errors++ } else { $childObjects = $cfgNode.SelectSingleNode("md:ChildObjects", $ns) if ($childObjects) { $found = $false foreach ($child in @($childObjects.ChildNodes)) { if ($child.NodeType -ne 'Element') { continue } if ($child.LocalName -eq $objType -and $child.InnerText.Trim() -eq $objName) { $found = $true if (-not $DryRun) { # Remove preceding whitespace if present $prev = $child.PreviousSibling if ($prev -and $prev.NodeType -eq 'Whitespace') { $childObjects.RemoveChild($prev) | Out-Null } $childObjects.RemoveChild($child) | Out-Null } Write-Host "[OK] Removed <$objType>$objName from ChildObjects" $actions++ break } } if (-not $found) { Write-Host "[WARN] <$objType>$objName not found in ChildObjects" } } # Save Configuration.xml if ($actions -gt 0 -and -not $DryRun) { $enc = New-Object System.Text.UTF8Encoding $true $sw = New-Object System.IO.StreamWriter($configXml, $false, $enc) $xmlDoc.Save($sw) $sw.Close() Write-Host "[OK] Configuration.xml saved" } } # --- 4. Remove from subsystem Content --- Write-Host "" Write-Host "--- Subsystems ---" $subsystemsDir = Join-Path $ConfigDir "Subsystems" $subsystemsFound = 0 $subsystemsCleaned = 0 function Remove-FromSubsystems { param([string]$dir) $xmlFiles = @(Get-ChildItem $dir -Filter "*.xml" -File -ErrorAction SilentlyContinue) foreach ($xmlFile in $xmlFiles) { $ssDoc = New-Object System.Xml.XmlDocument $ssDoc.PreserveWhitespace = $true try { $ssDoc.Load($xmlFile.FullName) } catch { continue } $ssNs = New-Object System.Xml.XmlNamespaceManager($ssDoc.NameTable) $ssNs.AddNamespace("md", "http://v8.1c.ru/8.3/MDClasses") $ssNs.AddNamespace("v8", "http://v8.1c.ru/8.1/data/core") $ssNode = $ssDoc.DocumentElement.SelectSingleNode("md:Subsystem", $ssNs) if (-not $ssNode) { continue } $propsNode = $ssNode.SelectSingleNode("md:Properties", $ssNs) if (-not $propsNode) { continue } $contentNode = $propsNode.SelectSingleNode("md:Content", $ssNs) if (-not $contentNode) { continue } $ssNameNode = $propsNode.SelectSingleNode("md:Name", $ssNs) $ssName = if ($ssNameNode) { $ssNameNode.InnerText } else { $xmlFile.BaseName } # Content items are Type.Name $targetRef = "${objType}.${objName}" $modified = $false foreach ($item in @($contentNode.ChildNodes)) { if ($item.NodeType -ne 'Element') { continue } $val = $item.InnerText.Trim() # Content format: "Subsystem.X" or "Catalog.X" etc. if ($val -eq $targetRef) { $script:subsystemsFound++ if (-not $DryRun) { $prev = $item.PreviousSibling if ($prev -and $prev.NodeType -eq 'Whitespace') { $contentNode.RemoveChild($prev) | Out-Null } $contentNode.RemoveChild($item) | Out-Null $modified = $true } Write-Host "[OK] Removed from subsystem '$ssName'" $script:subsystemsCleaned++ } } if ($modified -and -not $DryRun) { $enc = New-Object System.Text.UTF8Encoding $true $sw = New-Object System.IO.StreamWriter($xmlFile.FullName, $false, $enc) $ssDoc.Save($sw) $sw.Close() } # Recurse into child subsystems $childDir = Join-Path $dir ($xmlFile.BaseName) $childSubsystems = Join-Path $childDir "Subsystems" if (Test-Path $childSubsystems -PathType Container) { Remove-FromSubsystems -dir $childSubsystems } } } if (Test-Path $subsystemsDir -PathType Container) { Remove-FromSubsystems -dir $subsystemsDir if ($subsystemsCleaned -eq 0) { Write-Host "[OK] Not referenced in any subsystem" } } else { Write-Host "[OK] No Subsystems directory" } # --- 5. Delete object files --- Write-Host "" Write-Host "--- Files ---" if (-not $KeepFiles) { if ($hasDir -and -not $DryRun) { Remove-Item $objDir -Recurse -Force Write-Host "[OK] Deleted directory: $typePlural/$objName/" $actions++ } elseif ($hasDir) { Write-Host "[DRY] Would delete directory: $typePlural/$objName/" $actions++ } if ($hasXml -and -not $DryRun) { Remove-Item $objXml -Force Write-Host "[OK] Deleted file: $typePlural/$objName.xml" $actions++ } elseif ($hasXml) { Write-Host "[DRY] Would delete file: $typePlural/$objName.xml" $actions++ } if (-not $hasXml -and -not $hasDir) { Write-Host "[OK] No files to delete" } } else { Write-Host "[SKIP] File deletion skipped (-KeepFiles)" } # --- Summary --- Write-Host "" $totalActions = $actions + $subsystemsCleaned if ($DryRun) { Write-Host "=== Dry run complete: $totalActions actions would be performed ===" } else { Write-Host "=== Done: $totalActions actions performed ($subsystemsCleaned subsystem references removed) ===" } if ($errors -gt 0) { exit 1 } exit 0