Auto.ps1
#Set-StrictMode -Version Latest ##################################################### # Auto ##################################################### <#PSScriptInfo .VERSION 0.23 .GUID 602bc07e-a621-4738-8c27-0edf4a4cea8e .AUTHOR David Walker, Sitecore Dave, Radical Dave .COMPANYNAME David Walker, Sitecore Dave, Radical Dave .COPYRIGHT David Walker, Sitecore Dave, Radical Dave .TAGS sitecore powershell local install iis solr .LICENSEURI https://github.com/SitecoreDave/SharedSitecore.SitecoreLocal/blob/main/LICENSE .PROJECTURI https://github.com/SitecoreDave/SharedSitecore.SitecoreLocal .ICONURI .EXTERNALMODULEDEPENDENCIES .REQUIREDSCRIPTS .EXTERNALSCRIPTDEPENDENCIES .RELEASENOTES - see README.md #> <# .SYNOPSIS Auto All The Things! .DESCRIPTION PowerShell script that helps you Automate All The Things! .EXAMPLE PS> Auto 'name' PS> Auto az armtemplate.json .EXAMPLE PS> Auto 'name' 'template' .EXAMPLE PS> Auto 'name' 'template' 'd:\repos' .Link https://github.com/Radical-Dave/Auto .OUTPUTS System.String #> [CmdletBinding(SupportsShouldProcess=$true)] Param( [Parameter(Mandatory = $false, Position=0)] [string] $action = 'help', [Parameter(Mandatory = $false, Position=1)] [string] $data = '', [Parameter(Mandatory = $false, Position=2)] [string] $path = '', [string] $adApp = 'DevOps', [switch] $Force = $false, [switch] $recurse = $false ) begin { $Global:ErrorActionPreference = 'Stop' $PSScriptName = ($MyInvocation.MyCommand.Name.Replace(".ps1","")) $PSScriptVersion = (Test-ScriptFileInfo -Path $MyInvocation.MyCommand.Path | Select-Object -ExpandProperty Version) $PSCallingScript = if ($MyInvocation.PSCommandPath) { $MyInvocation.PSCommandPath | Split-Path -Parent } else { $null } Write-Verbose "#####################################################" Write-Host "# $PSScriptRoot/$PSScriptName $($PSScriptVersion):$action $data $path called by:$PSCallingScript" -ForegroundColor White $StopWatch = New-Object -TypeName System.Diagnostics.Stopwatch $StopWatch.Start() Write-Verbose "path:$path" if (@('az','tf') -contains $action -and $data -ne "") { if (Test-Path "$PSScriptRoot\data\$action\$data\tasks\$data.json") { $path = "$PSScriptRoot\data\$action\tasks.json"; } elseif (Test-Path "$PSScriptPath\data\$action\tasks\$data\$data.json") { $path = "$PSScriptPath\data\$action\$data\tasks.json"; } $tasks = $path } else { if (!$path -and $action -ne 'az') { if (Test-Path "$PSScriptRoot/$PSScriptName.json") { $path = "$PSScriptRoot/$PSScriptName.json" } else { $profileParent =Split-Path $profile -Parent if (Test-Path "$profileParent/$($PSScriptName).json") { $path = "$profileParent/$($PSScriptName).json" } } } } Write-Verbose "path:$path" #if (!(Test-Path $path)) { # throw "ERROR invalid path:$($path)" #} if ($path) { try { #$tasks = Get-Content .\auto.json | Out-String | Invoke-Expression $configFile = (Get-Content $path -Raw) | ConvertFrom-Json #Write-Host "config:$($config)" } catch { throw $_ } Write-Verbose "logs:$($configFile.logs)" try { #Write-Verbose "checking tasks" $tasksNode = $configFile.psobject.properties["tasks"].value #Write-Verbose "checking props" #Write-Host "tasksNode:$($tasksNode)" $tasks = $tasksNode.PSObject.Properties #Write-Host "tasks:$($tasks)" } catch { Write-Verbose "NON-Critical? Error parsing task(s): $_" } } if (!$path) { if ((Test-Path "$PSScriptPath\data\$action\$data\tasks.json")) { $path = "$PSScriptPath\data\$action\$data\tasks.json"; } } Write-Verbose "path:$path" } process { if ($action -eq 'add' -and $data.IndexOf('=') -gt -1) { Write-Host "add task:$data" $ds = $data.split('=') #if ($config.tasks["$data"]) { #if (!Force) { # throw "ERROR Task:$action already exists. Use -Force to overwrite." #} else { # $config.tasks | Add-Member -MemberType NoteProperty -Name "$($ds[0])" -Value "$($ds[1])" -PassThru -Force #} #} else { $configFile.tasks | Add-Member -MemberType NoteProperty -Name "$($ds[0])" -Value "$($ds[1])" -PassThru -Force #} $configFile | ConvertTo-Json | Out-File $path } elseif ($action -eq 'del' -or $action -eq 'delete' -and $data) { if (!$configFile -or !$configFile.tasks) { throw 'ERROR no config or config.tasks?' } Write-Host "delete task:$data" $configFile.tasks.PSObject.Properties.Remove("$data") $configFile | ConvertTo-Json | Out-File $path } else { $task = @() Write-Verbose "action:$action" if ($tasks -and $action -ne 'help') { $taskProperty = $tasks[$action] Write-Verbose "taskProperty:$taskProperty" if ($taskProperty) { $task = $taskProperty.value } } if ($task) { Write-Verbose "task:$task" if ($task -like '*$(data)*') { Write-Verbose "data:$data" $task = $task.replace('$(data)', "$($data)") #$task = $task -replace '$(data)', "$($data)" } Write-Verbose "task:$task" $cmd = $task Write-Verbose "cmd:$cmd" if ($cmd -like '~\*') { $cmd = (Join-Path (Split-Path $profile -Parent) ($task.Remove(0,2))) } elseif ($cmd -like '.\*') { $cmd = (Join-Path $PSScriptRoot ($task.Remove(0,2))) } elseif ($cmd.Substring(0,2) -eq '*\') { $name = $cmd.Remove(0,2) Write-Host "name:$name" $checkPath = $data if (!$checkPath) { #$cmd = (Join-Path $PSScriptRoot ($task.Remove(0,2))) $checkPath = "$(Split-Path $path -Parent)\data\tests\nested" } Write-Host "checkPath:$checkPath" $cmd = Get-ChildItem "$checkPath\$name*" -Recurse | Select-Object FullName if (!$cmd) { throw "ERROR could not find item $name in $checkPath -Recurse" } } Write-Host "Invoke:$cmd" -ForegroundColor White if (!$data) { Invoke-Expression -Command $cmd -OutVariable $results } else { Invoke-Expression -Command "$cmd $data" -OutVariable $results } Write-Host "RESULTS:$results" } else { if ($PSBoundParameters.ContainsKey('recurse')) { $filter = @('terraform') -contains $data ? "*.*" : "*.tf" #if (@('terraform') -contains $data) { $filter = "*.tf" Write-Host "action:$action" Write-Host "data:$data" Write-Host "filter:$filter" $cwd = $PWD #Get-ChildItem $data -recurse -exclude ".*" -include $filter | Foreach-Object { #Get-ChildItem $data -Directory -recurse -exclude ".*" -include $filter | Foreach-Object { #Get-ChildItem $data -recurse -exclude ".*" -include $filter | Select-Object Directory -Unique | Foreach-Object { $folders = Get-ChildItem -exclude ".*" -Path $data -Recurse | Where-Object {$_.PSIsContainer}# | ForEach-Object { foreach($folder in $folders) { if (Test-Path "$folder\$filter") { Write-Host "### X:$folder" Set-Location $folder -verbose #Write-Host "### Set:${$_}" #Write-Host "### Set:${$_.FullName}" Invoke-Expression -Command $action -OutVariable $results } } Write-Host "stop" Set-Location $cwd exit } #if ($action -ne 'help' -and $action -ne 'az' -and $action -ne 'tf') { if (@('help','az','tf','tfd','sql') -notcontains $action) { Write-Host "!#$! Task not found in $PSScriptName.json: $action, to add use: -addTask 'AutoScript'" -ForegroundColor White } if ($action -eq 'sql') { if(!$data) {$data = 'SELECT @@version'} #* FROM SYS.DATABASES #SYS_TABLES if ($path -notlike ';') { #use config $path = "Data Source=(local);Initial Catalog=.;Integrated Security=SSPI;" } Write-Host "RUN:$data" Write-Host "AGAINST:$path" Write-Host "path:$path" try { $connection = New-Object system.data.sqlclient.sqlconnection Write-Verbose "[BEGIN ] Creating the SQL Command object" $cmd = New-Object system.Data.SqlClient.SqlCommand $connection.connectionstring = $path $connection.open() #join the connection to the command object $cmd.connection = $connection $cmd.CommandText = $data Write-Verbose "[PROCESS] Invoking $data" if ($PSCmdlet.ShouldProcess($data)) { #determine what method to invoke based on the query Switch -regex ($data) { "^Select (\w+|\*)|(@@\w+ AS)" { $reader = $cmd.executereader() $out=@() #convert datarows to a custom object while ($reader.read()) { $h = [ordered]@{} for ($i=0;$i -lt $reader.FieldCount;$i++) { $col = $reader.getname($i) $h.add($col,$reader.getvalue($i)) } #for $out += new-object -TypeName psobject -Property $h } #while $out $reader.close() Break } "@@" { $cmd.ExecuteScalar() Break } Default { $cmd.ExecuteNonQuery() | Out-Null } } } } catch { throw "ERROR $PSScriptName sql $data - $_" #-InformationVariable results } } elseif (@('az','tf','tfd') -contains $action) { Write-Host "RUN:$path" $envPrefix = '' #if ($action -eq 'tf') { $envPrefix = 'TF_VAR_'} if (@('tf','tfd') -contains $action) { $envPrefix = 'TF_VAR_'} #todo: finish working with Nick to use set-env (otherwise use set-envs) @((Split-Path $profile -Parent),$PSScriptRoot,("$currLocation" -ne "$PSScriptRoot" ? $currLocation : ''),$data |