Почему я получаю сообщение об ошибке «Ошибка удаленного вызова процедуры» после остановки процесса Excel? - PullRequest
0 голосов
/ 30 апреля 2020

У меня есть следующий код, который преобразует листы Excel в CSV-файлы. Если файлы CSV не существуют / или уже существуют, но не используются (например, открыты в Excel), сценарий успешно генерирует файлы CSV (перезаписывая их, если они уже существуют)!

Однако, если файл CSV открывается в Excel, затем я получаю сообщение об ошибке «Не удается получить доступ к CSV-файла», который я определил, потому что он используется Excel (при открытии). Я знаю, что это на 100% причина, потому что если у меня есть существующий CSV-файл, открытый в блокноте, сценарий все равно перезаписывает CSV-файл, успешно запущенный.

, поэтому я попытался реализовать автоматическое разрешение c, которое составляет Get-Process 'exce[l]' | Stop-Process -Force, и хотя оно останавливает процесс (закрывает Excel), я получаю еще одну ошибку:

Convert-ExcelSheetsToCsv : Failed to save csv! Path: 'C:\Users\Documents\Folder1\CSV_Files\COS.csv'. The remote
procedure call failed. (Exception from HRESULT: 0x800706BE)

Convert-ExcelSheetsToCsv : Failed to save csv! Path: 'C:\Users\Documents\Folder1\CSV_Files\.csv'. The RPC server is
unavailable. (Exception from HRESULT: 0x800706BA)

После некоторых исследований я отключил свои надстройки COM-Excel, снова запустил сценарий, и все еще возникали исключения ...

com

Почему это так? ?

$currentDir = $PSScriptRoot

$csvPATH = Join-Path -Path $currentDir -ChildPath CSV_Files
New-Item -ItemType Directory -Force -Path $csvPATH | out-null

function Convert-ExcelSheetsToCsv {
    param(
        [Parameter(Mandatory, ValueFromPipelineByPropertyName, Position=1)]
        [ValidateNotNullOrEmpty()]
        [Alias('FullName')]
        [string]$Path,
        [Parameter(Mandatory = $false, Position=0)]
        [bool]$AppendFileName,
        [Parameter(Mandatory = $false, Position=2)]
        [bool]$ExcludeHiddenSheets,
        [Parameter(Mandatory = $false, Position=3)]
        [bool]$ExcludeHiddenColumns
    )
    Begin {
        $excel = New-Object -ComObject Excel.Application -Property @{
            Visible       = $false
            DisplayAlerts = $false
        }
    }
    Process {
        #$root = Split-Path -Path $Path
        $filename = [System.IO.Path]::GetFileNameWithoutExtension($Path)
        $workbook = $excel.Workbooks.Open($Path)

        foreach ($worksheet in ($workbook.Worksheets | Where { <# $_.Visible -eq -1 #> $_.Name -ne 'Security' -and $_.Name -ne 'Notes' })) {        
            if($ExcludeHiddenColumns) {
                $ColumnsCount = $worksheet.UsedRange.Columns.Count
                for ($i=1; $i -le $ColumnsCount; $i++)
                {
                    $column = $worksheet.Columns.Item($i).EntireColumn #$worksheet.sheets.columns.entirecolumn.hidden=$true
                    if ($column.hidden -eq $true)
                    {   
                        $columnname = $column.cells.item(1,$i).value2

                        if ($worksheet.Visible -eq 0) #worksheet hidden
                        {
                            "`r`nHidden column [{0}] found in hidden [{1}] worksheet. Deleting..." -f $columnname, $($worksheet.name)
                        }
                        else {
                            "`r`nHidden column [{0}] found in [{1}] worksheet. Deleting..." -f $columnname, $($worksheet.name)
                        }

                        try {
                            $column.Delete() | out-null

                            "`r`nHidden column [{0}] was Deleted! Proceeding with Export to CSV operation...`r`n" -f $columnname
                        }
                        catch {
                            Write-Error -Message "`r`nFailed to Delete hidden column [$columnname] from [$($worksheet.name)] worksheet! $PSItem"
                            #$_ | Select *
                        }

                        #$i = $i - 1
                    }
                }
            }

            if ($ExcludeHiddenSheets) {
                if ($worksheet.Visible -eq -1) #worksheet visible
                {
                    $ws = $worksheet
                }
            }
            else {
                $ws = $worksheet
            }

            if ($AppendFileName) {
                $name = Join-Path -Path $csvPATH <# $root #> -ChildPath "${filename}_$($ws.Name).csv"
            }
            else {
                $name = Join-Path -Path $csvPATH <# $root #> -ChildPath "$($ws.Name).csv"
            }

            try {
                $ws.SaveAs($name, 6) #6 to ignore formatting and convert to pure text, otherwise, file could end up containing rubbish
            } 
            catch {
                if ($error[0].ToString().Contains("Cannot access"))
                {
                    "`r`n'{0}' is currently in use.`r`n Attempting to override usage by trying to stop Excel process..." -f $name

                    try {
                        #Only 'excel' will be matched, but because a wildcard [] is used, not finding a match will not generate an error.
                        #https://stackoverflow.com/a/32475836/8397835

                        Get-Process 'exce[l]' | Stop-Process -Force

                        "`r`nExcel process stopped! Saving '{0}' ..." -f $name

                        $ws.SaveAs($name, 6)
                    }
                    catch {
                        Write-Error -Message "Failed to save csv! Path: '$name'. $PSItem"
                    }
                }
                else {
                    Write-Error -Message "Failed to save csv! Path: '$name'. $PSItem"
                }
            }
        }
    }
    End {
        $excel.Quit()
        $null = [System.Runtime.InteropServices.Marshal]::ReleaseComObject($excel)
    }
}

Get-ChildItem -Path $currentDir -Filter *.xlsx | Convert-ExcelSheetsToCsv -AppendFileName 0 -ExcludeHiddenSheets 1 -ExcludeHiddenColumns 1 #0 for false, so that filename of excel file isnt appended, and only sheet names are the names of the csv files

1 Ответ

0 голосов
/ 01 мая 2020

Это потому, что объект Excel тоже уничтожается. правильный способ сделать это - завершить процесс ДО создания экземпляра объекта Excel:

Begin {
    Get-Process 'exce[l]' | Stop-Process -Force
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...