mirror of
https://github.com/Nikolay-Shirokov/cc-1c-skills.git
synced 2026-06-26 15:04:34 +03:00
260 lines
12 KiB
PowerShell
260 lines
12 KiB
PowerShell
# help-add v1.7 — Add built-in help to 1C object
|
|
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
|
|
param(
|
|
[Parameter(Mandatory)]
|
|
[string]$ObjectName,
|
|
|
|
[string]$Lang = "ru",
|
|
|
|
[string]$SrcDir = "src"
|
|
)
|
|
|
|
$ErrorActionPreference = "Stop"
|
|
[Console]::OutputEncoding = [System.Text.Encoding]::UTF8
|
|
[Console]::InputEncoding = [System.Text.Encoding]::UTF8
|
|
|
|
# --- Support guard (Ext/ParentConfigurations.bin) ---
|
|
# See docs/1c-support-state-spec.md. Blocks edits of vendor objects "на замке" /
|
|
# read-only configs unless allowed. Trigger = bin present; reaction from
|
|
# .v8-project.json editingAllowedCheck (deny|warn|off, default deny). Never
|
|
# throws — guard errors degrade to allow.
|
|
function Get-RootUuid([string]$xmlPath) {
|
|
if (-not (Test-Path $xmlPath)) { return $null }
|
|
try {
|
|
[xml]$mx = Get-Content -Path $xmlPath -Encoding UTF8
|
|
$el = $mx.DocumentElement.FirstChild
|
|
while ($el -and $el.NodeType -ne 'Element') { $el = $el.NextSibling }
|
|
if ($el) { $u = $el.GetAttribute("uuid"); if ($u) { return $u } }
|
|
} catch {}
|
|
return $null
|
|
}
|
|
function Find-V8Project([string]$startDir) {
|
|
$d = $startDir
|
|
for ($i = 0; $i -lt 20 -and $d; $i++) {
|
|
$pj = Join-Path $d ".v8-project.json"
|
|
if (Test-Path $pj) { return $pj }
|
|
$parent = [System.IO.Path]::GetDirectoryName($d)
|
|
if ($parent -eq $d) { break }
|
|
$d = $parent
|
|
}
|
|
return $null
|
|
}
|
|
function Get-EditMode([string]$cfgDir) {
|
|
try {
|
|
$pj = Find-V8Project (Get-Location).Path
|
|
if (-not $pj) { $pj = Find-V8Project $cfgDir }
|
|
if (-not $pj) { return 'deny' }
|
|
$proj = Get-Content -Raw $pj | ConvertFrom-Json
|
|
$cfgFull = [System.IO.Path]::GetFullPath($cfgDir).TrimEnd('\', '/')
|
|
if ($proj.databases) {
|
|
foreach ($db in $proj.databases) {
|
|
if ($db.configSrc) {
|
|
$src = [System.IO.Path]::GetFullPath($db.configSrc).TrimEnd('\', '/')
|
|
if ($cfgFull -eq $src -or $cfgFull.StartsWith($src + [System.IO.Path]::DirectorySeparatorChar)) {
|
|
if ($db.editingAllowedCheck) { return $db.editingAllowedCheck }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if ($proj.editingAllowedCheck) { return $proj.editingAllowedCheck }
|
|
return 'deny'
|
|
} catch { return 'deny' }
|
|
}
|
|
function Assert-EditAllowed([string]$targetPath, [string]$require) {
|
|
try {
|
|
$rp = $targetPath
|
|
try { $rp = (Resolve-Path $targetPath -ErrorAction Stop).Path } catch {}
|
|
$elemUuid = Get-RootUuid $rp
|
|
$cfgDir = $null; $binPath = $null
|
|
$d = if (Test-Path $rp -PathType Container) { $rp } else { [System.IO.Path]::GetDirectoryName($rp) }
|
|
for ($i = 0; $i -lt 12 -and $d; $i++) {
|
|
if (-not $elemUuid) { $elemUuid = Get-RootUuid "$d.xml" }
|
|
if (-not $cfgDir) {
|
|
$cand = Join-Path (Join-Path $d "Ext") "ParentConfigurations.bin"
|
|
if ((Test-Path $cand) -or (Test-Path (Join-Path $d "Configuration.xml"))) { $cfgDir = $d; $binPath = $cand }
|
|
}
|
|
if ($elemUuid -and $cfgDir) { break }
|
|
$parent = [System.IO.Path]::GetDirectoryName($d)
|
|
if ($parent -eq $d) { break }
|
|
$d = $parent
|
|
}
|
|
# New object (no element file): fall back to config root uuid.
|
|
if (-not $elemUuid -and $cfgDir) { $elemUuid = Get-RootUuid (Join-Path $cfgDir "Configuration.xml") }
|
|
if (-not $binPath -or -not (Test-Path $binPath)) { return }
|
|
$bytes = [System.IO.File]::ReadAllBytes($binPath)
|
|
if ($bytes.Length -le 32) { return }
|
|
$start = 0
|
|
if ($bytes.Length -ge 3 -and $bytes[0] -eq 0xEF -and $bytes[1] -eq 0xBB -and $bytes[2] -eq 0xBF) { $start = 3 }
|
|
$text = [System.Text.Encoding]::UTF8.GetString($bytes, $start, $bytes.Length - $start)
|
|
$hm = [regex]::Match($text, '^\{6,(\d+),(\d+),')
|
|
if (-not $hm.Success) { return }
|
|
$G = [int]$hm.Groups[1].Value
|
|
$K = [int]$hm.Groups[2].Value
|
|
if ($K -eq 0) { return }
|
|
$best = $null
|
|
if ($elemUuid) {
|
|
$u = [regex]::Escape($elemUuid.ToLower())
|
|
foreach ($m in [regex]::Matches($text, "([0-2]),0,$u")) {
|
|
$f1 = [int]$m.Groups[1].Value
|
|
if ($null -eq $best -or $f1 -lt $best) { $best = $f1 }
|
|
}
|
|
}
|
|
$blocked = $false; $code = ""; $reason = ""
|
|
if ($G -eq 1) { $blocked = $true; $code = "capability-off"; $reason = "возможность изменения конфигурации выключена (вся конфигурация read-only)" }
|
|
elseif ($require -eq 'removed') {
|
|
if ($null -ne $best -and $best -ne 2) { $blocked = $true; $code = "not-removed"; $reason = "объект не снят с поддержки — удаление сломает обновления" }
|
|
}
|
|
else {
|
|
if ($null -ne $best -and $best -eq 0) { $blocked = $true; $code = "locked"; $reason = "объект на замке — редактирование сломает обновления" }
|
|
}
|
|
if (-not $blocked) { return }
|
|
$mode = Get-EditMode $cfgDir
|
|
if ($mode -eq 'off') { return }
|
|
# Use Console.Error (not Write-Error) — under ErrorActionPreference=Stop the
|
|
# latter throws and would be swallowed by this function's own catch.
|
|
if ($mode -eq 'warn') { [Console]::Error.WriteLine("[support-guard] ПРЕДУПРЕЖДЕНИЕ: $reason. Цель: $rp"); return }
|
|
$head = "[support-guard] Редактирование отклонено: это объект типовой конфигурации на поддержке поставщика, прямое редактирование молча сломает будущие обновления."
|
|
$cfe = "Рекомендуемый путь: внести доработку в расширение (навыки cfe-borrow / cfe-patch-method) — состояние поддержки менять не нужно, обновления вендора сохраняются."
|
|
$offNote = "Снять проверку для этой базы: editingAllowedCheck = warn|off в .v8-project.json."
|
|
if ($code -eq "capability-off") {
|
|
$state = "Состояние: у всей конфигурации выключена возможность изменения (режим read-only «из коробки») — поэтому объект «$rp» редактировать нельзя."
|
|
$fix = "Либо снять защиту явно (навык support-edit, два шага):`n 1. support-edit -Path ""$cfgDir"" -Capability on — включить возможность изменения (объекты пока остаются на замке);`n 2. support-edit -Path ""$rp"" -Set editable — открыть этот объект для редактирования.`n Изменение применяется в базу полной загрузкой выгрузки и обходит механизм обновлений вендора."
|
|
} elseif ($code -eq "not-removed") {
|
|
$state = "Состояние: объект «$rp» на поддержке (не снят с поддержки) — его удаление разорвёт обновления вендора."
|
|
$fix = "Либо сначала снять объект с поддержки, затем удалять:`n support-edit -Path ""$rp"" -Set off-support — объект уходит из-под обновлений, после этого удаление безопасно."
|
|
} else {
|
|
$state = "Состояние: объект «$rp» на замке (возможность изменения конфигурации включена, но сам объект не редактируется)."
|
|
$fix = "Либо разрешить редактирование этого объекта (навык support-edit, выбрать одно):`n support-edit -Path ""$rp"" -Set editable — редактировать и дальше получать обновления вендора (возможны конфликты слияния);`n support-edit -Path ""$rp"" -Set off-support — снять с поддержки: обновления по объекту больше не приходят."
|
|
}
|
|
[Console]::Error.WriteLine("$head`n$state`n$cfe`n$fix`n$offNote")
|
|
exit 1
|
|
} catch { return }
|
|
}
|
|
|
|
# --- Detect format version ---
|
|
|
|
function Detect-FormatVersion([string]$dir) {
|
|
$d = $dir
|
|
while ($d) {
|
|
$cfgPath = Join-Path $d "Configuration.xml"
|
|
if (Test-Path $cfgPath) {
|
|
$content = [System.IO.File]::ReadAllText($cfgPath, [System.Text.Encoding]::UTF8)
|
|
$head = $content.Substring(0, [Math]::Min(2000, $content.Length))
|
|
if ($head -match '<MetaDataObject[^>]+version="(\d+\.\d+)"') { return $Matches[1] }
|
|
}
|
|
$parent = Split-Path $d -Parent
|
|
if ($parent -eq $d) { break }
|
|
$d = $parent
|
|
}
|
|
return "2.17"
|
|
}
|
|
|
|
$formatVersion = Detect-FormatVersion (Resolve-Path $SrcDir).Path
|
|
|
|
# --- Проверки ---
|
|
|
|
$objectDir = Join-Path $SrcDir $ObjectName
|
|
$extDir = Join-Path $objectDir "Ext"
|
|
|
|
if (-not (Test-Path $extDir)) {
|
|
Write-Error "Каталог объекта не найден: $extDir. Проверьте путь ObjectName (например Catalogs/МойСправочник)."
|
|
exit 1
|
|
}
|
|
|
|
$helpXmlPath = Join-Path $extDir "Help.xml"
|
|
if (Test-Path $helpXmlPath) {
|
|
Write-Error "Справка уже существует: $helpXmlPath"
|
|
exit 1
|
|
}
|
|
|
|
Assert-EditAllowed $objectDir 'editable'
|
|
|
|
# --- Кодировка ---
|
|
|
|
$encBom = New-Object System.Text.UTF8Encoding($true)
|
|
|
|
# --- 1. Help.xml ---
|
|
|
|
$helpXml = @"
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
<Help xmlns="http://v8.1c.ru/8.3/xcf/extrnprops" xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="$formatVersion">
|
|
<Page>$Lang</Page>
|
|
</Help>
|
|
"@
|
|
|
|
[System.IO.File]::WriteAllText($helpXmlPath, $helpXml, $encBom)
|
|
|
|
# --- 2. Help/<lang>.html ---
|
|
|
|
$helpDir = Join-Path $extDir "Help"
|
|
New-Item -ItemType Directory -Path $helpDir -Force | Out-Null
|
|
|
|
$helpHtmlPath = Join-Path $helpDir "$Lang.html"
|
|
|
|
$helpHtml = @"
|
|
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
|
|
<html>
|
|
<head>
|
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
|
|
<link rel="stylesheet" type="text/css" href="v8help://service_book/service_style"/>
|
|
</head>
|
|
<body>
|
|
<h1>$ObjectName</h1>
|
|
<p>Описание.</p>
|
|
</body>
|
|
</html>
|
|
"@
|
|
|
|
[System.IO.File]::WriteAllText($helpHtmlPath, $helpHtml, $encBom)
|
|
|
|
# --- 3. Проверка IncludeHelpInContents в метаданных форм ---
|
|
|
|
$formsDir = Join-Path $objectDir "Forms"
|
|
if (Test-Path $formsDir) {
|
|
$formMetaFiles = Get-ChildItem -Path $formsDir -Filter "*.xml" -File
|
|
foreach ($formMeta in $formMetaFiles) {
|
|
$xmlDoc = New-Object System.Xml.XmlDocument
|
|
$xmlDoc.PreserveWhitespace = $true
|
|
$xmlDoc.Load($formMeta.FullName)
|
|
|
|
$nsMgr = New-Object System.Xml.XmlNamespaceManager($xmlDoc.NameTable)
|
|
$nsMgr.AddNamespace("md", "http://v8.1c.ru/8.3/MDClasses")
|
|
|
|
$includeHelp = $xmlDoc.SelectSingleNode("//md:IncludeHelpInContents", $nsMgr)
|
|
if (-not $includeHelp) {
|
|
# Добавить после <FormType>
|
|
$formType = $xmlDoc.SelectSingleNode("//md:FormType", $nsMgr)
|
|
if ($formType) {
|
|
$newElem = $xmlDoc.CreateElement("IncludeHelpInContents", "http://v8.1c.ru/8.3/MDClasses")
|
|
$newElem.InnerText = "false"
|
|
$parent = $formType.ParentNode
|
|
$nextSibling = $formType.NextSibling
|
|
# Вставить перенос + табуляцию + элемент
|
|
$ws = $xmlDoc.CreateWhitespace("`n`t`t`t")
|
|
if ($nextSibling) {
|
|
$parent.InsertBefore($ws, $nextSibling) | Out-Null
|
|
$parent.InsertBefore($newElem, $ws) | Out-Null
|
|
} else {
|
|
$parent.AppendChild($ws) | Out-Null
|
|
$parent.AppendChild($newElem) | Out-Null
|
|
}
|
|
|
|
$settings = New-Object System.Xml.XmlWriterSettings
|
|
$settings.Encoding = $encBom
|
|
$settings.Indent = $false
|
|
$stream = New-Object System.IO.FileStream($formMeta.FullName, [System.IO.FileMode]::Create)
|
|
$writer = [System.Xml.XmlWriter]::Create($stream, $settings)
|
|
$xmlDoc.Save($writer)
|
|
$writer.Close()
|
|
$stream.Close()
|
|
|
|
Write-Host " IncludeHelpInContents добавлен: $($formMeta.Name)"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Write-Host "[OK] Создана справка: $ObjectName"
|
|
Write-Host " Метаданные: $helpXmlPath"
|
|
Write-Host " Страница: $helpHtmlPath"
|