我需要将 ID 列表拉入 powershell 参数验证集,如下所示:
function Do-Stuff {
[Cmdletbinding()]
param(
[ValidateSet("Seattle","NewYork","London","Atlanta" )]
[String]$Site
)
我不想手动指定集合中的城市,而是想从一个已经列出所有城市的现有 xml 文档中提取。加载后,站点名称出现在 $xml.var.sites.id 中。这是可能的,而且可能更重要的是,这是一个好主意吗?
最佳答案
最近的一个项目不得不这样做。没有意识到枚举有多么容易,谢谢 mjolinor!
另一种方法是使用动态参数。使用 Get-Help 查找帮助:
#Look for the Dynamic Parameters section in here
Get-Help about_Functions_Advanced_Parameters
其他资源:
当前函数定义:
Function New-DynamicParam {
<#
.SYNOPSIS
Helper function to simplify creating dynamic parameters
.DESCRIPTION
Helper function to simplify creating dynamic parameters
Example use cases:
Include parameters only if your environment dictates it
Include parameters depending on the value of a user-specified parameter
Provide tab completion and intellisense for parameters, depending on the environment
Please keep in mind that all dynamic parameters you create will not have corresponding variables created.
One of the examples illustrates a generic method for populating appropriate variables from dynamic parameters
Alternatively, manually reference $PSBoundParameters for the dynamic parameter value
.NOTES
Credit to http://jrich523.wordpress.com/2013/05/30/powershell-simple-way-to-add-dynamic-parameters-to-advanced-function/
Added logic to make option set optional
Added logic to add RuntimeDefinedParameter to existing DPDictionary
Added a little comment based help
.PARAMETER Name
Name of the dynamic parameter
.PARAMETER ValidateSet
If specified, set the ValidateSet attribute of this dynamic parameter
.PARAMETER Mandatory
If specified, set the Mandatory attribute for this dynamic parameter
.PARAMETER ParameterSetName
If specified, set the ParameterSet attribute for this dynamic parameter
.PARAMETER Position
If specified, set the Position attribute for this dynamic parameter
.PARAMETER ValueFromPipelineByPropertyName
If specified, set the ValueFromPipelineByPropertyName attribute for this dynamic parameter
.PARAMETER HelpMessage
If specified, set the HelpMessage for this dynamic parameter
.PARAMETER DPDictionary
If specified, add resulting RuntimeDefinedParameter to an existing RuntimeDefinedParameterDictionary (appropriate for multiple dynamic parameters)
If not specified, create and return a RuntimeDefinedParameterDictionary (appropriate for a single dynamic parameter)
.EXAMPLE
function Show-Free
{
[CmdletBinding()]
Param()
DynamicParam {
$options = @( gwmi win32_volume | %{$_.driveletter} | sort )
New-DynamicParam -Name Drive -ValidateSet $options -Position 0 -Mandatory
}
begin{
#have to manually populate
$drive = $PSBoundParameters.drive
}
process{
$vol = gwmi win32_volume -Filter "driveletter='$drive'"
"{0:N2}% free on {1}" -f ($vol.Capacity / $vol.FreeSpace),$drive
}
} #Show-Free
Show-Free -Drive <tab>
# This example illustrates the use of New-DynamicParam to create a single dynamic parameter
# The Drive parameter ValidateSet populates with all available volumes on the computer for handy tab completion / intellisense
.EXAMPLE
# I found many cases where I needed to add many dynamic parameters
# The DPDictionary parameter lets you specify an existing dictionary
# The block of code in the Begin block loops through bound parameters and defines variables if they don't exist
Function Test-DynPar{
[cmdletbinding()]
param(
[string[]]$x = $Null
)
DynamicParam
{
#Create the RuntimeDefinedParameterDictionary
$Dictionary = New-Object System.Management.Automation.RuntimeDefinedParameterDictionary
New-DynamicParam -Name AlwaysParam -options @( gwmi win32_volume | %{$_.driveletter} | sort ) -DPDictionary $Dictionary
#Add dynamic parameters to $dictionary
if($x -eq 1)
{
New-DynamicParam -Name X1Param1 -Options 1,2 -mandatory -DPDictionary $Dictionary
New-DynamicParam -Name X1Param2 -DPDictionary $Dictionary
New-DynamicParam -Name X3Param3 -DPDictionary $Dictionary
}
else
{
New-DynamicParam -Name OtherParam1 -mandatory -DPDictionary $Dictionary
New-DynamicParam -Name OtherParam2 -DPDictionary $Dictionary
New-DynamicParam -Name OtherParam3 -DPDictionary $Dictionary
}
#return RuntimeDefinedParameterDictionary
$Dictionary
}
Begin
{
#This standard block of code loops through bound parameters...
#If no corresponding variable exists, one is created
foreach($param in $PSBoundParameters.Keys)
{
if (-not ( Get-Variable -name $param -scope 0 -ErrorAction SilentlyContinue ) )
{
New-Variable -Name $Param -Value $PSBoundParameters.$param
Write-Verbose "Adding variable for dynamic parameter '$param' with value '$($PSBoundParameters.$param)'"
}
}
#Appropriate variables should now be defined and accessible
Get-Variable -scope 0
}
}
# This example illustrates the creation of many dynamic parameters using New-DynamicParam
# You must create a RuntimeDefinedParameterDictionary object ($dictionary here)
# To each New-DynamicParam call, add the -DPDictionary parameter pointing to this RuntimeDefinedParameterDictionary
# At the end of the DynamicParam block, return the RuntimeDefinedParameterDictionary
# Initialize all bound parameters using the provided block or similar code
#>
param(
[string]
$Name,
[string[]]
$ValidateSet,
[switch]
$Mandatory,
[string]
$ParameterSetName="__AllParameterSets",
[int]
$Position,
[switch]
$ValueFromPipelineByPropertyName,
[string]
$HelpMessage,
[validatescript({
if(-not ( $_ -is [System.Management.Automation.RuntimeDefinedParameterDictionary] -or -not $_) )
{
Throw "DPDictionary must be a System.Management.Automation.RuntimeDefinedParameterDictionary object, or not exist"
}
$True
})]
$DPDictionary = $false
)
#Create attribute object, add attributes, add to collection
$ParamAttr = New-Object System.Management.Automation.ParameterAttribute
$ParamAttr.ParameterSetName = $ParameterSetName
if($mandatory)
{
$ParamAttr.Mandatory = $True
}
if($Position -ne $null)
{
$ParamAttr.Position=$Position
}
if($ValueFromPipelineByPropertyName)
{
$ParamAttr.ValueFromPipelineByPropertyName = $True
}
if($HelpMessage)
{
$ParamAttr.HelpMessage = $HelpMessage
}
$AttributeCollection = New-Object 'Collections.ObjectModel.Collection[System.Attribute]'
$AttributeCollection.Add($ParamAttr)
#param validation set if specified
if($ValidateSet)
{
$ParamOptions = New-Object System.Management.Automation.ValidateSetAttribute -ArgumentList $ValidateSet
$AttributeCollection.Add($ParamOptions)
}
#Create the dynamic parameter
$Parameter = New-Object -TypeName System.Management.Automation.RuntimeDefinedParameter -ArgumentList @($Name, [string], $AttributeCollection)
#Add the dynamic parameter to an existing dynamic parameter dictionary, or create the dictionary and add it
if($DPDictionary)
{
$DPDictionary.Add($Name, $Parameter)
}
else
{
$Dictionary = New-Object System.Management.Automation.RuntimeDefinedParameterDictionary
$Dictionary.Add($Name, $Parameter)
$Dictionary
}
}
使用此函数的示例:
#Function has already been added to this session...
function Do-Stuff {
[cmdletbinding()]
param()
DynamicParam
{
#Example borrowing from TheMadTechnician:
#New-DynamicParam -Name Site -ValidateSet $(([xml](gc c:\temp\config.xml)).getElementsByTagName("City").Name) -Mandatory
#I don't have that file... simplification
New-DynamicParam -Name Site -ValidateSet "Seattle", "NewYork", "London", "Atlanta" -Mandatory
}
Begin
{
#This standard block of code loops through bound parameters...
#If no corresponding variable exists, one is created
foreach($param in $PSBoundParameters.Keys)
{
if (-not ( Get-Variable -name $param -scope 0 -ErrorAction SilentlyContinue ) )
{
New-Variable -Name $param -Value $PSBoundParameters.$param
Write-Verbose "Adding variable for dynamic parameter '$param' with value '$($PSBoundParameters.$param)'"
}
}
$Site
}
}
最后,最终结果:
再举一个有一些其他动态参数的例子
Function New-LabMachine
{
[cmdletbinding()]
param(
[ValidateSet("ADDSNew","ADDSExisting")]
[string[]]
$Role
)
DynamicParam
{
#Define dynamicparam dictionary. Create a hashtable for splatting params
$Dictionary = New-Object System.Management.Automation.RuntimeDefinedParameterDictionary
$dict = @{DPDictionary = $Dictionary}
#Add dynamic parameters to populate and validate Lab names
$Labs = "Lab1", "Lab2" #@( Get-Lab | Select -ExpandProperty LabName -ErrorAction SilentlyContinue )
New-DynamicParam -Name LabName -Mandatory -ValidateSet $Labs @dict
if($Role -contains 'ADDSNew')
{
#AD Forest info
New-DynamicParam -Name DomainName -Mandatory @dict -HelpMessage "Provide domain name for first domain in forest"
New-DynamicParam -Name ForestMode -Mandatory -ValidateSet "Win2008","Win2008R2","Win2012","Win2012R2" @dict
}
if($Role -contains 'ADDSExisting')
{
New-DynamicParam -Name DomainName -Mandatory @dict
New-DynamicParam -Name username -Mandatory @dict
New-DynamicParam -Name password -Mandatory @dict
}
#Return the dictionary for dynamic params
$Dictionary
}
Begin
{
#This standard block of code loops through bound parameters...
#If no corresponding variable exists, one is created
foreach($param in $PSBoundParameters.Keys )
{
if (-not ( Get-Variable -name $param -scope 0 -ErrorAction SilentlyContinue ) -and "Verbose", "Debug" -notcontains $param )
{
New-Variable -Name $Param -Value $PSBoundParameters.$param -Description DynParam
Write-Verbose "Adding variable for dynamic parameter '$param' with value '$($PSBoundParameters.$param)'"
}
}
#Display the bound parameters
$PSBoundParameters.keys | ForEach-Object {Get-Variable -Name $_}
}
}
结果:
在我看来,这些对最终用户非常有帮助。我通常使用它们来提供与您的目标类似的 IntelliSense 和制表符完成支持。只要它们为您提供的值(value)超过它们的轻微开销和一点点额外的复杂性,它们就值得:)
我为文字墙道歉!干杯!
关于xml - 从 XML 填充参数列表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22993799/
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere
是否有类似“RVMuse1”或“RVMuselist[0]”之类的内容而不是键入整个版本号。在任何时候,我们都会看到一个可能包含5个或更多ruby的列表,我们可以轻松地键入一个数字而不是X.X.X。这也有助于rvmgemset。 最佳答案 这在RVM2.0中是可能的=>https://docs.google.com/document/d/1xW9GeEpLOWPcddDg_hOPvK4oeLxJmU3Q5FiCNT7nTAc/edit?usp=sharing-知道链接的任何人都可以发表评论
我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"
我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)
两者都可以defsetup(options={})options.reverse_merge:size=>25,:velocity=>10end和defsetup(options={}){:size=>25,:velocity=>10}.merge(options)end在方法的参数中分配默认值。问题是:哪个更好?您更愿意使用哪一个?在性能、代码可读性或其他方面有什么不同吗?编辑:我无意中添加了bang(!)...并不是要询问nobang方法与bang方法之间的区别 最佳答案 我倾向于使用reverse_merge方法:option
我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano
我有一个驼峰式字符串,例如:JustAString。我想按照以下规则形成长度为4的字符串:抓取所有大写字母;如果超过4个大写字母,只保留前4个;如果少于4个大写字母,则将最后大写字母后的字母大写并添加字母,直到长度变为4。以下是可能发生的3种情况:ThisIsMyString将产生TIMS(大写字母);ThisIsOneVeryLongString将产生TIOV(前4个大写字母);MyString将生成MSTR(大写字母+tr大写)。我设法用这个片段解决了前两种情况:str.scan(/[A-Z]/).first(4).join但是,我不太确定如何最好地修改上面的代码片段以处理最后一种