PowerShell 技能连载 - 远程执行代码

在一个域环境中,PowerShell 远程操作功能几乎是开箱即用的。您所需要做的知识在目标机器上启用远程功能(从 Server 2012 开始,PowerShell 远程操作功能对于 Administrators 组用户缺省是启用的)。

在 PowerShell 3.0 中,需要人为地启用远程功能,这就是一切要做的事了(需要管理员权限):

PS> Enable-PSRemoting -SkipNetworkProfileCheck -Force

你不需要在客户端(准备发送命令的机器)上配置任何东西。

下一步,任何管理员可以将命令发送到启用了远程操作功能的机器上去执行它。以下例子将列出目标机器上所有和 PowerShell 相关的进程:

$code =
{
      Get-Process -Name powershell*, wsmprovhost -ErrorAction SilentlyContinue
}

$list = 'server1', 'w2k12-niki', 'pc11box'
Invoke-Command -ScriptBlock $code #-ComputerName $list

当您原样执行这段代码的时候,Invoke-Command 在您自己的机器上运行存储在 $code 中的代码块。

它列出所有运行中的 PowerShell 控制台的实例、ISE PowerShell 编辑器,以及所有由您机器上别人初始化的 PowerShell 隐藏远程会话。

而当您去掉 -ComputerName 参数的注释,代码将会在 $list 变量存储的所有计算机上执行。请确保它们存在并且已启用了远程操作功能。当您从远程计算机收到数据时,PowerShell 自动在返回的信息上附加一个 "PSComputerName" 属性,用来存储返回信息的计算机名。

评论

PowerShell 技能连载 - 访问所有用户的桌面

Resolve-Path 是一个相当棒的查找相同深度路径用的 Cmdlet。例如,以下是一段很短小的代码,它在您机器的每个用户桌面上创建一个文本文件:

$root = Split-Path $env:USERPROFILE

Resolve-Path $root\*\Desktop |
  ForEach-Object {
    $Path = Join-Path -Path $_ -ChildPath 'hello there.txt'
    'Here is some content...' | Out-File -FilePath $Path
    Write-Warning "Creating $Path"
  }

以管理员权限运行您的脚本,它将在您机器中所有用户的桌面上创建一个文件:

WARNING: Creating C:\Users\Administrator\Desktop\hello there.txt
WARNING: Creating C:\Users\CustomerService\Desktop\hello there.txt
WARNING: Creating C:\Users\Guest\Desktop\hello there.txt
WARNING: Creating C:\Users\PSTestGer\Desktop\hello there.txt
WARNING: Creating C:\Users\Tester\Desktop\hello there.txt
WARNING: Creating C:\Users\Tobias\Desktop\hello there.txt
评论

PowerShell 技能连载 - 在 ISE 中快速选中结果

如果您想快速地选中并复制 PowerShell 3.0 ISE 编辑器中控制台窗格的结果到您的博客或您喜欢的文字处理器中,例如将结果保存为文档或者贴到文章中。以下是实现的方法:

当您运行一个命令后,光标位于 ISE 控制台窗格中并且等待输入新的命令。只需要按住 CTRL 并按下 键。光标将向左移动一位。然后按下 CTRL+A 选取所有的结果,然后按下 CTRL+C 将它们复制到剪贴板。完成!

评论

PowerShell 技能连载 - 加速多个 WMI 查询

当您远程执行 Get-WmiObject 命令时,它将会创建一个新的连接。所以如果您查询不同的 WMI 类时,每个查询会使用一个不同的连接,这样将会影响总体性能。

从 PowerShell 3.0 开始,有一些列新的 Cmdlet。使用这些 Cmdlet 可以容易地使用现有的连接高效地运行多个查询:

$session = New-CimSession –ComputerName localhost
$os = Get-CimInstance –ClassName Win32_OperatingSystem –CimSession $session
$bios = Get-CimInstance -ClassName Win32_BIOS -CimSession $session

会话缺省使用 WSMAN 协议:

PS> $session

Id           : 1
Name         : CimSession1
InstanceId   : 0bb38128-3633-4eb8-8b55-6d2910b89bcd
ComputerName : localhost
Protocol     : WSMAN

当您创建会话是,您可以显式地指定一个不同的远程传输协议,例如 DCOM。

评论

PowerShell 技能连载 - 访问终极 PowerShell 生存指南

您熟悉 PowerShell 生存指南吗?它是终极的 PowerShell 维基页面,包括视频、Module,和您很可能需要的关于 PowerShell 的一切资源。最棒的部分:它是一个开放式维基,所以如果您自己创建了一个 PowerShell 资源,希望让其他人知道它,那么您完全可以编辑 PowerShell 生存指南来让别人知道它。

http://social.technet.microsoft.com/wiki/contents/articles/183.windows-powershell-survival-guide.aspx

评论

修正 SubManager 的一个 bug

目前最好的字幕下载工具是爱页工作室SubManager 字幕管理器。程序通过射手网开放的API自批量动下载电影对应的字幕,省去逐个找字幕的麻烦。其工作原理与射手播放器一致,但有十余项增强。

SubManager 字幕管理器

目前它的最新版是 v2013.7.2,我在使用过程中发现一个问题:

当搜索目录中含有扩展名为空的文件时,程序出现异常:

有关调用实时(JIT)调试而不是此对话框的详细信息,
请参见此消息的结尾。

************** 异常文本 **************
System.ArgumentException: 字符串的长度不能为零。
参数名: oldValue
   在 System.String.Replace(String oldValue, String newValue)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir, String[] fileList)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir, String[] fileList)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir, String[] fileList)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir, String[] fileList)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir, String[] fileList)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir, String[] fileList)
   在 ShooterDownloader.DownloadForm.PopulateFileList(String dir)
   在 ShooterDownloader.DownloadForm.txtDir_KeyUp(Object sender, KeyEventArgs e)
   在 System.Windows.Forms.Control.OnKeyUp(KeyEventArgs e)
   在 System.Windows.Forms.Control.ProcessKeyEventArgs(Message& m)
   在 System.Windows.Forms.Control.ProcessKeyMessage(Message& m)
   在 System.Windows.Forms.Control.WmKeyChar(Message& m)
   在 System.Windows.Forms.Control.WndProc(Message& m)
   在 System.Windows.Forms.TextBoxBase.WndProc(Message& m)
   在 System.Windows.Forms.TextBox.WndProc(Message& m)
   在 System.Windows.Forms.Control.ControlNativeWindow.OnMessage(Message& m)
   在 System.Windows.Forms.Control.ControlNativeWindow.WndProc(Message& m)
   在 System.Windows.Forms.NativeWindow.Callback(IntPtr hWnd, Int32 msg, IntPtr wparam, IntPtr lparam)

Reflector 将 SubManager.exe 反编译,得到它的源代码,定位到 DownloadFormPopulateFileList(String, String[]) : Void 方法。其中只有一行用到 String.Replace() 方法:

if ((File.Exists(info.FullName.Replace(info.Extension, ".srt")) || File.Exists(info.FullName.Replace(info.Extension, ".ass"))) || (File.Exists(info.FullName.Replace(info.Extension, ".ssa")) || File.Exists(info.FullName.Replace(info.Extension, ".smi"))))

把它改为:

if (string.IsNullOrEmpty(info.Extension) || (File.Exists(info.FullName.Replace(info.Extension, ".srt")) || File.Exists(info.FullName.Replace(info.Extension, ".ass"))) || (File.Exists(info.FullName.Replace(info.Extension, ".ssa")) || File.Exists(info.FullName.Replace(info.Extension, ".smi"))))

并重新编译,该错误提示消失了。

我联系了作者 ayeah,他表示将在下一个版本修复。

评论

PowerShell 技能连载 - 获取 WMI 智能感知信息

Get-WmiObject 并未对 WMI 类提供智能感知信息,所以您要么事先知道 WMI 类的名字,要么使用 -List 参数来搜索它。

不过有一个聪明的技巧:Get-CimInstance 命令几乎完成相同的事情,并且它的参数 -ClassName 也接受一个 WMI 类名。而这个参数提供了智能感知支持。

在 PowerShell 3.0 ISE 中进行以下操作:

PS> Get-CimInstance -ClassName Win32_ 

然后按下 CTRL+SPACE 键来调用智能感知。请观察状态栏提示。由于有几百个 WMI 类名,所以首次尝试的时候,智能感知在获取所有类信息的时候可能会超时。过一段时间,或者您稍微限定以下类名,它就可以正常工作了。

所以只要用 Get-CimInstance 来代替 Get-WmiObject,然后在智能感知的支持下选择类名,然后将 Cmdlet 和参数改回 Get-WmiObject -Class 即可。

或者,从头到尾都使用 Get-CimInstance。它返回基本相同价值的信息。但在缺省情况下,它使用 WSMan 协议来进行远程操作,而不是 DCOM。

评论

PowerShell 技能连载 - 列出“真实”的硬盘

WMI 可以提供一个系统的很多信息,但是有些时候这些信息太多了。当您查询逻辑磁盘时,您得到的往往不止是物理磁盘。

设置额外的过滤器可以解决此问题。以下这行代码通过设置 DriveType=3 来获取物理驱动器:

PS> Get-WmiObject -Class Win32_LogicalDisk -Filter 'DriveType=3'

DeviceID     : C:
DriveType    : 3
ProviderName : 
FreeSpace    : 4468535296
(...)

由于 Get-WmiObject 有一个 -ComputerName 参数,所以您也可以远程获取该信息。如果您想知道有哪些其他的驱动器类型,只需要去掉过滤条件,或者用搜索引擎搜索 "Win32_LogicalDisk DriveType"

评论

PowerShell 技能连载 - 改进版的自动加载 Module

当您按照上一个技巧进行设置以后,PowerShell 3.0 便可以自动加载 Module。然而,对于某些 Module,该技术没有效果。那些 Cmdlet 仍然只能通过 Import-Module 的方式来导入 Module。

导致它们的原因是由于它们构建的方式。PowerShell 无法检测到这些 Module 导出了哪些 Cmdlet。

然而,有一行简单的命令可以让更多的 Module 自动变得可用:

PS> Get-Module -ListAvailable -Refresh

-Refresh switch 参数告知 PowerShell 完整地遍历所有的 Module 并且生成或刷新内部的命令缓存。

评论

PowerShell 技能连载 - 自动加载 Module

从 PowerShell 3.0 开始,PowerShell 具备了能够智能识别哪些 Cmdlet 是由哪个扩展 Module 导出的特性。所以您再也不需要知道 Module 的名称并且(用 Import-Module 手动导入它)。与之相反,自动完成和智能感知特性将为所有安装在标准 Module 文件夹中的每一个命令提供建议。以下是列出这些标准文件夹的方法:

PS> $env:PSModulePath -split ';'
C:\Users\Victor\Documents\WindowsPowerShell\Modules
C:\Program Files\WindowsPowerShell\Modules
C:\Windows\system32\WindowsPowerShell\v1.0\Modules\

这些标准文件夹也许不一定相同,您可以根据需要增加更多的文件夹到环境变量中,甚至将 Module 存放在 USB 闪存盘或外置驱动器中。以下命令将把 USB 驱动器的路径增加到您的模块列表中,这样在该文件夹中的所有 Module 也将会被自动加载:

PS> $env:PSModulePath += ';g:\mypersonalmodules'
评论