Se você quer a versão resumida, simples e resumida, a USMT é "apenas" uma ferramenta de backup. Eu sou bastante novo com isso, mas quando script, pode se tornar uma ferramenta valiosa para o administrador. Espero que isso ajude!
O seguinte, que é um arquivo em lote, automatiza a parte carregada da USMT. Eu não tenho o código em vigor para o array_compare.ps1, mas se alguém estiver realmente interessado, posso publicá-lo. Divirta-se e deixe-me saber se você tem alguma dúvida!
Verificando se o script está sendo executado em um estado elevado e, se não, se eleva.
:::::::::::::::::::::::::::::::::::::::::
::
:: Automatically check & get admin rights
:::::::::::::::::::::::::::::::::::::::::
@echo off
CLS
ECHO.
ECHO =============================
ECHO Running Admin shell
ECHO =============================
: checkPrivileges
NET FILE 1>NUL 2>NUL
if '%errorlevel%' == '0' ( goto gotPrivileges ) else ( goto getPrivileges )
:getPrivileges
if '%1'=='ELEV'
(shift & goto gotPrivileges)
ECHO.
ECHO **************************************
ECHO Invoking UAC for Privilege Escalation
ECHO **************************************
setlocal DisableDelayedExpansion
set "batchPath=%~0"
setlocal EnableDelayedExpansion
ECHO Set UAC = CreateObject^("Shell.Application"^) > "%temp%\OEgetPrivileges.vbs"
ECHO UAC.ShellExecute "!batchPath!", "ELEV", "", "runas", 1 >> "%temp%\OEgetPrivileges.vbs"
"%temp%\OEgetPrivileges.vbs"
exit /B
:gotPrivileges
::::::::::::::::::::::::::::
::START::::::::::::::::::::::::::::
setlocal & pushd .
REM Run shell as admin (example)
::::::::::::::::::::::::::::
Mapeamento de uma unidade de rede que usei para armazenar e manter diversos arquivos necessários. Ele também armazenou os arquivos USMT para serem carregados em um pouco.
@ECHO OFF
START /wait NET USE z: \NETWORK SHARE\FOLDER\FOLDER
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
:: Lines 42 - 44 Run the array_compare.ps1 as Admin ::
:: to run a search to find the new user name and output ::
:: it to a file that we will read from shortly. ::
:: THIS CAN BE PROVIDED...LET ME KNOW!! ::
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
O array_compare.ps1, pega duas listas de nomes de usuários e compara-os, envia o arquivo CSV que será lido para importar os perfis.
SET ThisScriptsDirectory=%~dp0
SET PowerShellScriptPath="Z:\Path\to\powershell\comparison\array_compare.ps1"
PowerShell -NoProfile -ExecutionPolicy Bypass -Command "& {Start-Process PowerShell -ArgumentList '->NoProfile -ExecutionPolicy Bypass -File ""%PowerShellScriptPath%""' -Verb RunAs}";
::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
:: Change the Powershell ExecutionPolicy so that scripts will run without issue. ::
::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
PowerShell -NoProfile -ExecutionPolicy Unrestricted -Command "& {Start-Process PowerShell -ArgumentList '-NoProfile -ExecutionPolicy Unrestricted'}"
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
:: ***Declaring our variables*** ::
:: Location is the folder name where we stored the scanstate information. ::
:: Domain2 is the new domain that we are migrating to. ::
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
Scanstate salva os arquivos em um compartilhamento de rede em uma pasta com o nome do host. Aqui somos solicitados a inserir o nome do host do dispositivo que está sendo substituído.
set /P location= Please enter the OLD computer name:
Como o USMT tem dois .exe diferentes baseados no nível de bit do sistema operacional, Win 7x86 vs Win 7x64, eu queria executar uma verificação e, com base nisso, executar o estado de carga adequado.
Interrogando o processador para determinar o tipo de arquitetura.
:CheckOS
if /i %processor_architecture%==AMD64 GOTO AMD64
Else
if /i %processor_architecture%==x86 GOTO x86
Else
if /i %PROCESSOR_ARCHITEW6432%==AMD64 GOTO AMD64
:AMD64
Z:
CD Z:\Program Files\USMT\x64\
xcopy *.* /e /v /y c:\windows\usmt\
C:
CD c:\windows\usmt\
setlocal EnableDelayedExpansion
Set inputfile=Z:\%location%\NAME_OF_FILE_THAT_CONTAINS_BOTH_USERIDS.txt
for /f "tokens=1-2" %%A in (%inputfile%) do call :USMTx64 %%A%%B
:USMTx64
ECHO Syncing %1 ...
loadstate Z:\%location% /c /lac /lae /i:miguser.xml /i:migapp.xml /i:novell_excl.xml /config:config.xml /ue:Administrator /ue:Admin* /ue:EO-* /ue:*\* /ui:%1 /mu:<OLDDOMAIN>\%1:ID\%2 /mu:%1:ID\%2 /v:13
if NOT ["%errorlevel%"]==["0"] (
GOTO END
)
exit /b
:x86
Z:
CD Z:\Program Files\USMT\x86
xcopy *.* /e /v /y c:\windows\usmt\
C:
CD c:\windows\usmt\
Set inputfile=Z:\%location%\NAME_OF_FILE_THAT_CONTAINS_BOTH_USERIDS.txt
Criado um loop para / f que lê o arquivo de saída do array_compare.ps1 anterior. Isso é o que realmente automatiza a USMT para nós. Ele passará pelo número X de vezes, cada vez carregando um novo usuário.
for /f "tokens=1-2" %%A in (%inputfile%) do call :USMTx86 %%A%%B
:USMTx86
ECHO Syncing %1 ...
loadstate Z:\%location% /c /lac /lae /i:miguser.xml /i:migapp.xml /i:novell_excl.xml /config:config.xml /ue:Administrator /ue:EO-* /ue:Admin* /ui:%1 /ue:*\* /mu:<OLDDOMAIN>\%1:ID\%2 /mu:%1:ID\%2 /v:13
::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: ::::::::::::
:: Checks for errors in the USMT commands.
:: If found exits the CALL sub and continues with the script.
:::: SOURCE: https://coderwall.com/p/jexjlw
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
Esse loop de erro foi absolutamente necessário, pelo menos para mim! Quando o for / f chega à última iteração, a USMT irá cometer erros. Se isso acontecer e ainda houver comandos a serem executados, o arquivo em lote falhará. Isso lida com isso, sai do loop for / f e continua.
if NOT ["%errorlevel%"]==["0"] (
GOTO END
)
exit /b
:END
:: Deletes the mapped network drive
net use z: /delete /yes
Exit