Запустите команду оболочки с аргументами из сценария PowerShell.

StackOverflow https://stackoverflow.com/questions/2479434

  •  21-09-2019
  •  | 
  •  

Вопрос

Мне нужно извлечь и сохранить несколько таблиц из удаленной базы данных SQL с помощью bcp.Я хотел бы написать сценарий PowerShell для вызова bcp для каждой таблицы и сохранения данных.На данный момент у меня есть скрипт, который создает необходимые аргументы для bcp.Однако я не могу понять, как передать аргументы в bcp.Каждый раз, когда я запускаю скрипт, вместо этого он просто показывает справку по bcp.Должно быть, это что-то очень простое, чего я не получаю.

#commands bcp database.dbo.tablename out c:\temp\users.txt -N -t, -U uname -P pwd -S <servername>
$bcp_path = "C:\Program Files\Microsoft SQL Server\90\Tools\Binn\bcp.exe"
$serverinfo =@{}
$serverinfo.add('table','database.dbo.tablename')
$serverinfo.add('uid','uname')
$serverinfo.add('pwd','pwd')
$serverinfo.add('server','servername')
$out_path= "c:\Temp\db\"
$args = "$($serverinfo['table']) out $($out_path)test.dat -N -t, -U $($serverinfo['uid']) -P $($serverinfo['pwd']) -S $($serverinfo['server'])"

#this is the part I can't figure out
& $bcp_path $args
Это было полезно?

Решение

Прежде всего, $args — автоматическая переменная;вы не можете установить его, поэтому любая строка типа $args = foo ничего не сделает (даже при включенном строгом режиме;хотя жалобу было бы неплохо).

Тогда вы передаете программе только один аргумент (строку).I содержит пробелы, но они правильно экранированы или заключены в круглые скобки, поэтому программа видит только один аргумент.

Вам нужно будет использовать массив для аргументов программы, а не одну строку, если вы хотите заранее сохранить его в переменной.И вам нужно назвать это как-то иначе, чем $args:

$arguments = "$($serverinfo['table'])",
             'out',"$($out_path)test.dat",
             '-N','-t,',
             '-U',"$($serverinfo['uid'])",
             '-P',"$($serverinfo['pwd'])",
             '-S',"$($serverinfo['server'])"

& $bcp_path $arguments

Или, что я бы предпочел, на самом деле, вы можете просто написать это в одной строке, что избавит вас от большей части уродства:

$out_path = 'c:\Temp\db'
& $bcp_path $serverinfo['table'] out $out_path\test.dat -N '-t,' -U $serverinfo['uid'] -P $serverinfo['pwd'] -S $serverinfo['server']

Другие советы

Некоторые приложения командной строки, которым необходимо принимать сумасшедшие аргументы в стиле Каннам с косыми чертами, кавычками, двойными кавычками, равенствами, двоеточиями, тире — настоящий коктейль.

PowerShell, по моему опыту, иногда просто не справляется.Поэтому я записываю файл .cmd и выполняю его из cmd.exe, вот так:

echo $("Running command: " + $commandLine);

$rnd = $(([string](Get-Random -Minimum 10000 -Maximum 99999999)) + ".cmd");
$commandFilePath = $(Join-Path -Path $env:TEMP -ChildPath $rnd);
echo $commandLine | Out-File -FilePath $commandFilePath -Encoding ascii;

& cmd.exe /c $commandFilePath

Убедитесь, что вы выводите данные в формате ASCII, так как Юникод по умолчанию может некорректно работать с cmd.exe (он рявкнул на меня и при первой попытке показал странные символы).

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top