关于c#调用PowerShell来控制SCVMM,网上有很多例子,也比较简单,但创建虚拟机的过程,是一个很漫长的时间,所以一般来说,创建的时候都希望可以实时的显示当前虚拟机的创建进度。当时这个问题困扰了我了一段时间,但找到方法以后,发现其实很简单。

环境:

  Win server 2008 R2 + Hyper-v + SCVMM2008 R2

目的:

  C#调PowerShell在SCVMM中创建虚拟机时,实时显示创建进度

在SCVMM2008R2中手动创建一个vm(虚拟机)时,作业界面中会显示很详细的创建进度,包括有哪些子任务,每个任务的完成度、状态等信息。SCVMM的界面操作是基于Powershell的,所以肯定有ps脚本可以实现上述目的。

microsoft提供的创建虚拟机的ps脚本中,提到如下内容(为显示PS脚本中部分内容被回车)

$NewVMTasks = [System.Array]::CreateInstance("Microsoft.SystemCenter.VirtualMachineManager.Task"$NumVMs)$NewVMs = [System.Array]::CreateInstance("Microsoft.SystemCenter.VirtualMachineManager.VM"$NumVMs)$i = 0# Loop that creates each VM asynchronously.while($NumVMs -gt 0){# Generate a unique VM name.$VMRnd = $Random.next()$NewVMName = $VMName+$VMRnd# Get the ratings for each host and sort the hosts by ratings.$Ratings = @(Get-VMHostRating -Template $Template -VMHost $VMHosts -DiskSpaceGB $DiskSizeGB  -VMName $NewVMName | where { $_.Rating -gt 0| Sort-Object -property Rating -descending)if ($Ratings.Count -gt 0    {$VMHost = $Ratings[0].VMHost$VMPath = $Ratings[0].VMHost.VMPaths[0]# Create a new VM from the template and add an additional VHD# to the VM.$NewVMJobGroup = [System.Guid]::NewGuid()$VMAdditionalVhd | Add-VirtualHardDisk -Bus 0 -Lun 1 -IDE -JobGroup $NewVMJobGroup$NewVMs[$i= New-VM -Template $Template -Name $NewVMName -Description $NewVMName  -VMHost $VMHost -Path $VMPath -RunAsynchronously -StartVM -JobGroup $NewVMJobGroup$NewVMTasks[$i= $NewVMs[$i].MostRecentTask$i = $i + 1   }$NumVMs = $NumVMs - 1   Start-Sleep -seconds 5}

脚本中的Task,其实就是我们所需要的,可以实时显示创建进度的类(Microsoft.SystemCenter.VirtualMachineManager.Task)。

也就是说在通过c#调用powershell创建虚拟时,如果可以得到此类的实例,即可解决问题。

在SCVMM提供的ps脚本中,创建虚机的cmdlet为“new-vm .....”后面带了很多参数,比如-name(虚机名字),-path(存储路径),-jobGroup(作业ID,作业ID一般通过 [System.Guid]::NewGuid()获得,唯一标示),-jobVariable(记录当前作业实例)等等,其中的-jobVariable即是我们所需要的。

关于-jobVariable这个参数的详细信息,及实例可以参考微软帮助文档

http://technet.microsoft.com/en-us/library/bb963731.aspx

例如下面的解释很详细了,即调用new-vm命令后,会将当前作业存在-jobvariable 后面指定的变量中。

##################################################################### Run a VMM cmdlet that creates a job - in this example script, the # cmdlet is New-V2V, so the job is the creation of a new VM from an # existing VMware VM.####################################################################$VM = New-V2V -VMXPath $LegacyVM -VMHost $VMHost -Name $VMName  -Path $VMPath -Memory $Memory -Runasynchronously -Jobvariable "Job"##################################################################### Track the status of the running job.####################################################################$JobNameString = $Job.CmdletName+" "+$Job.ResultName# Loop while the job is running, writing progress using current step # and progress values from the job.while ($job.status -eq "Running"Write-Progress -Activity "$JobNameString" -Status $Job.CurrentStep  -PercentComplete $Job.ProgressValue; Start-Sleep -seconds 5}

-jobvariable 一般跟-Runasynchronously参数一起使用,因为你获得了job变量当然希望ps异步执行了,

在c#中如何使用-jobvariable参数?请参考下面代码中红色部分

private void CreateVM(Host host           , Template template           , HardwareProfile hardwareProfile           , Runspace runspace){  Command newVM = new Command("new-vm");    newVM.Parameters.Add("vmhost", host);    newVM.Parameters.Add("Template", template);    newVM.Parameters.Add("hardwareprofile", hardwareProfile);    newVM.Parameters.Add("name", newVMName);    newVM.Parameters.Add("Description"            , string.Format("vm created by {0} on {1}"                      , Environment.UserName, DateTime.Now));  newVM.Parameters.Add("path", host.VMPaths[0]);  newVM.Parameters.Add("owner"this.owner);   newVM.Parameters.Add("startvm"true);  newVM.Parameters.Add("jobgroup"this.jobGroupID);  newVM.Parameters.Add("jobvariable""newvmtask");  newVM.Parameters.Add("runAsynchronously"true);    using (Pipeline pipeline = GetCommandPipe(runspace))  {    pipeline.Commands.Add(newVM);       Collection<PSObject> result = pipeline.Invoke();       pipeline.Stop();Task task = runspace.SessionStateProxy.GetVariable("newvmtask"as Task;task.Updated += new EventHandler<ObjectChangedEventArgs>(task_Updated);   }}private void task_Updated(object sender, ObjectChangedEventArgs e)  if (OnTaskProcess != null)    {        Task taskTemp = e.Object as Task;this.OnTaskProcess(taskTemp);     }}

上述代码中,很多东西不全,只是介绍如何取到-jobvariable参数指定的Task实例,在Command中添加-jobvariable

//Task对象放于名为newvmtask的变量中,此处随便定义。newVM.Parameters.Add("jobvariable""newvmtask");newVM.Parameters.Add("runAsynchronously"true);//指定cmdlet异步执行

在异步执行完后,在Runspace的Session中可以取到之前指定的Task实例,(此处为newvmtask)

Task task = runspace.SessionStateProxy.GetVariable("newvmtask") as Task;
此处的newvmtask是依据在command中-jobvariable参数指定的实例名。

取到Task对象实例后,即可通过其update事件,来实时显示Task的处理进度。

task.Updated += new EventHandler<ObjectChangedEventArgs>(task_Updated);

在task_Updated中可以作自己想做的事情。

Microsoft.SystemCenter.VirtualMachineManager.Task

Task类中存在一个属性Steps,类型为List<Step>,可以通过此属性,浏览一个作业的子过程的运行状况。

Step为Microsoft.SystemCenter.VirtualMachineManager.Step类

顺便提一下,SCVMM2008R2中powershell的Snapin为"Microsoft.SystemCenter.VirtualMachineManager"。

C#调PowerShell在SCVMM中创建虚拟机时,实时显示创建进度的更多相关文章

  1. OpenStack Horizon创建虚拟机时增加虚拟机OS用户

    背景 通过OpenStack的Horizon使用镜像创建虚拟机(以Ubuntu为例),如果不知道镜像的用户名和密码,在创建好虚拟机之后,无法登录虚拟机的OS.因此,我们需要一种方法,创建虚拟机时增加用 ...

  2. Unity3d中使用摄像机制作实时显示小地图

    Unity3d中使用摄像机制作实时显示小地图,以之前的tank为例.开始制作之前场景中物体如图. 开始制作,步骤1:新建一个camera及一个plane.对齐位置,将camera改名为camera_U ...

  3. Android中如何下载文件并显示下载进度

    原文地址:http://jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/1125/2057.html 这里主要讨论三种方式:AsyncTask.Serv ...

  4. Android 创建虚拟机时“提示no system images installed for this target”

    经上网查证,发现原因在于CPU/ABI选项无法选择,并显示“No system images installed for this target”,也就是没有适合的系统镜像,通过与安装好了的ADT-b ...

  5. 创建虚拟机时,提示No valid host was found解决办法

    1.http://blog.csdn.net/yxwmzouzou/article/details/43892261 2.http://www.cnblogs.com/kevingrace/p/601 ...

  6. vagrant 创建虚拟机时遇到问题

    问题1 :  ceph-node3: Warning: Authentication failure. Retrying.. 问题分析: ssh 认证失败,在向虚拟机拷贝内容时权限不足. 解决办法: ...

  7. Azure 基础:使用 powershell 创建虚拟网络

    什么是虚拟网络 虚拟网络是您的网络在 Azure 云上的表示形式.您可以完全控制虚拟网络的 IP 地址.DNS 的设置.安全策略和路由表.您还可以更进一步,把虚拟网络划分为多个子网.然后用它们连接您的 ...

  8. VS中 无法创建虚拟目录 本地IIS IIS Express 外部主机

    从前就有个疑问了,为什么我拉取别人写好的代码后就可以在IIS里面生成一个网站呢? 这里所谓的生成网站,是指包含了所有源代码文件的网站:相对地,发布网站,就是指包含被编译的源文件所得到的DLL文件的网站 ...

  9. Azure 中的虚拟网络和虚拟机

    创建 Azure 虚拟机 (VM) 时,必须创建虚拟网络 (VNet) 或使用现有的 VNet. 此外,还需要确定如何在 VNet 上访问 VM. 在创建资源之前必须做好规划,确保了解网络资源的限制. ...

随机推荐

  1. [CSP-S模拟测试]:春思(数学)

    蝶恋花·春景花褪残红青杏小.燕子飞时,绿水人家绕.枝上柳绵吹又少.天涯何处无芳草!墙里秋千墙外道.墙外行人,墙里佳人笑.笑渐不闻声渐悄.多情却被无情恼.(本词是伤春之作,写春景清新秀丽.同时,景中又有 ...

  2. 一片关于Bootstarp4的文章

    一.Bootstarp Bootstrap 是全球最受欢迎的前端组件库,用于开发响应式布局.移动设备优先的 WEB 项目.可以让你快速的排版,不用在写那些繁杂的样式.Bootstrap是开源免费的,设 ...

  3. C#-Newtonsoft.Json遍历并修改JSON

    遍历 JObject:https://www.newtonsoft.com/json/help/html/JObjectProperties.htm 遍历 JArray: string json = ...

  4. 小白学 Python 爬虫(26):为啥上海二手房你都买不起

    人生苦短,我用 Python 前文传送门: 小白学 Python 爬虫(1):开篇 小白学 Python 爬虫(2):前置准备(一)基本类库的安装 小白学 Python 爬虫(3):前置准备(二)Li ...

  5. zabbix配置-模板

    1.配置=>模板=>创建模板=>输入模板名称和群组 2.配置=>模板=>找到刚刚创建的模板=>点击应用集(applications)=>创建应用集=>输 ...

  6. pip安装第三方库报错Retrying (Retry(total=1, connect=None, read=None, redirect=None, status=None))...

    pip安装第三方库时报错Retrying (Retry(total=1, connect=None, read=None, redirect=None, status=None))...,详细报错见下 ...

  7. 记录js中的兼容问题及解决办法

    1.获取非行内样式的兼容问题: 2.获取事件对象的兼容问题: 3.事件冒泡的兼容: 4.keyCode的兼容问题: 5.处理默认事件的兼容问题: 6.事件的绑定兼容问题:

  8. 原生js事件绑定

    一.JS事件 (一)JS事件分类 1.鼠标事件: click/dbclick/mouseover/mouseout 2.HTML事件: onload/onunload/onsubmit/onresiz ...

  9. Java内存模型之happens-before原则

    我们无法就所有场景来规定某个线程修改的变量何时对其他线程可见,但是我们可以指定某些规则,这规则就是happens-before,从JDK 5 开始,JMM就使用happens-before的概念来阐述 ...

  10. Java中的HashMap的2种遍历方式比较

    首先我们准备数据,准备一个map Map<String, String> map = new HashMap<String, String>(); for (int i = 0 ...