Skip to content

Commit

Permalink
Add initial bootstrap script (#2669)
Browse files Browse the repository at this point in the history
  • Loading branch information
BernieWhite authored Dec 14, 2024
1 parent f3417dc commit 6e50b7e
Show file tree
Hide file tree
Showing 3 changed files with 325 additions and 1 deletion.
2 changes: 1 addition & 1 deletion docs/scenarios/containers/container-execution.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ The below example will use the latest PowerShell image released and install the
# Copyright (c) Microsoft Corporation.
# Licensed under the MIT License.

FROM mcr.microsoft.com/powershell:7.2-ubuntu-22.04
FROM mcr.microsoft.com/powershell:7.4-ubuntu-22.04
SHELL ["pwsh", "-command"]

RUN Install-Module -Name 'PSRule','PSRule.Rules.Azure' -Force
Expand Down
3 changes: 3 additions & 0 deletions src/bootstrap/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
# CI Bootstrap script

This directory contains a common bootstrap script that can be used to run PSRule in a CI/CD pipeline.
321 changes: 321 additions & 0 deletions src/bootstrap/bootstrap.ps1
Original file line number Diff line number Diff line change
@@ -0,0 +1,321 @@
# Copyright (c) Microsoft Corporation.
# Licensed under the MIT License.

#
# PSRule bootstrap script.
#

# See details at: https://github.com/microsoft/PSRule/blob/main/src/bootstrap/bootstrap.ps1

[CmdletBinding()]
param (
# The root path for analysis
[Parameter(Mandatory = $False)]
[String]$Path = $Env:INPUT_PATH,

# The type of input
[Parameter(Mandatory = $False)]
[ValidateSet('repository', 'inputPath')]
[String]$InputType = $Env:INPUT_INPUTTYPE,

# The path to input files
[Parameter(Mandatory = $False)]
[String]$InputPath = $Env:INPUT_INPUTPATH,

# The path to find rules
[Parameter(Mandatory = $False)]
[String]$Source = $Env:INPUT_SOURCE,

# Rule modules to use
[Parameter(Mandatory = $False)]
[String]$Modules = $Env:INPUT_MODULES,

# A baseline to use
[Parameter(Mandatory = $False)]
[String]$Baseline = $env:INPUT_BASELINE,

# The conventions to use
[Parameter(Mandatory = $False)]
[String]$Conventions = $Env:INPUT_CONVENTIONS,

# The path to an options file.
[Parameter(Mandatory = $False)]
[String]$Option = $Env:INPUT_OPTION,

# Filters output to include results with the specified outcome.
[Parameter(Mandatory = $False)]
[String]$Outcome = $Env:INPUT_OUTCOME,

# The output format
[Parameter(Mandatory = $False)]
[ValidateSet('None', 'Yaml', 'Json', 'NUnit3', 'Csv', 'Markdown', 'Sarif')]
[String]$OutputFormat = $Env:INPUT_OUTPUTFORMAT,

# The path to store formatted output
[Parameter(Mandatory = $False)]
[String]$OutputPath = $Env:INPUT_OUTPUTPATH,

# Determine if a pre-release module version is installed.
[Parameter(Mandatory = $False)]
[String]$PreRelease = $Env:INPUT_PRERELEASE,

# The name of the PowerShell repository where PSRule modules are installed from.
[String]$Repository = $Env:INPUT_REPOSITORY,

# Determines if a job summary is written.
[Parameter(Mandatory = $False)]
[String]$Summary = $Env:INPUT_SUMMARY,

# The specific version of PSRule to use.
[Parameter(Mandatory = $False)]
[String]$Version = $Env:INPUT_VERSION
)

$workspacePath = $Env:GITHUB_WORKSPACE;
$ProgressPreference = [System.Management.Automation.ActionPreference]::SilentlyContinue;
if ($Env:SYSTEM_DEBUG -eq 'true') {
$VerbosePreference = [System.Management.Automation.ActionPreference]::Continue;
}

# Check inputType
if ([String]::IsNullOrEmpty($InputType) -or $InputType -notin 'repository', 'inputPath') {
$InputType = 'repository';
}

# Set workspace
if ([String]::IsNullOrEmpty($workspacePath)) {
$workspacePath = $PWD;
}

# Set Path
if ([String]::IsNullOrEmpty($Path)) {
$Path = $workspacePath;
}
else {
$Path = Join-Path -Path $workspacePath -ChildPath $Path;
}

# Set InputPath
if ([String]::IsNullOrEmpty($InputPath)) {
$InputPath = $Path;
}
else {
$InputPath = Join-Path -Path $Path -ChildPath $InputPath;
}

# Set Source
if ([String]::IsNullOrWhiteSpace($Source)) {
$Source = [System.IO.Path]::GetFullPath([System.IO.Path]::Combine($Path, '.ps-rule/'))
}
elseif (![System.IO.Path]::IsPathFullyQualified($Source)) {
$Source = [System.IO.Path]::GetFullPath([System.IO.Path]::Combine($Path, $Source))
}

# Set conventions
if (![String]::IsNullOrEmpty($Conventions)) {
$Conventions = @($Conventions.Split(',', [System.StringSplitOptions]::RemoveEmptyEntries) | ForEach-Object {
$_.Trim();
});
}
else {
$Conventions = @();
}

# Set repository
if ([String]::IsNullOrEmpty($Repository)) {
$Repository = 'PSGallery'
}

function WriteDebug {
[CmdletBinding()]
param (
[Parameter(Position = 0, Mandatory = $True)]
[String]$Message
)
process {
if ($Env:SYSTEM_DEBUG -eq 'true' -or $Env:ACTIONS_STEP_DEBUG -eq 'true') {
Write-Host "::debug::$Message";
}
}
}

#
# Check and install modules
#
$dependencyFile = Join-Path -Path $Env:GITHUB_ACTION_PATH -ChildPath 'modules.json';
$latestVersion = (Get-Content -Path $dependencyFile -Raw | ConvertFrom-Json -AsHashtable -Depth 5).dependencies.PSRule.version;
$checkParams = @{
RequiredVersion = $latestVersion
}
if (![String]::IsNullOrEmpty($Version)) {
$checkParams['RequiredVersion'] = $Version;
if ($PreRelease -eq 'true') {
$checkParams['AllowPrerelease'] = $True;
}
}

# Look for existing versions of PSRule
Write-Host "[info] Using repository: $Repository";
$installed = @(Get-InstalledModule -Name PSRule @checkParams -ErrorAction Ignore)
if ($installed.Length -eq 0) {
Write-Host "[info] Installing PSRule: $($checkParams.RequiredVersion)";
$Null = Install-Module -Repository $Repository -Name PSRule @checkParams -Scope CurrentUser -Force;
}
foreach ($m in $installed) {
Write-Host "[info] Using existing module $($m.Name): $($m.Version)";
}

# Look for existing modules
Write-Host '';
$moduleNames = @()
if (![String]::IsNullOrEmpty($Modules)) {
$moduleNames = $Modules.Split(',', [System.StringSplitOptions]::RemoveEmptyEntries);
}
$moduleParams = @{
Scope = 'CurrentUser'
Force = $True
}
if ($PreRelease -eq 'true') {
$moduleParams['AllowPrerelease'] = $True;
}

# Install each module if not already installed
foreach ($m in $moduleNames) {
$m = $m.Trim();
Write-Host "> Checking module: $m";
try {
if ($Null -eq (Get-InstalledModule -Name $m -ErrorAction Ignore)) {
Write-Host ' - Installing module';
$Null = Install-Module -Repository $Repository -Name $m @moduleParams -AllowClobber -ErrorAction Stop;
}
else {
Write-Host ' - Already installed';
}
# Check
if ($Null -eq (Get-InstalledModule -Name $m)) {
Write-Host " - Failed to install";
}
else {
Write-Host " - Using version: $((Get-InstalledModule -Name $m).Version)";
}
}
catch {
Write-Host "::error::An error occurred installing a dependency module '$m'. $($_.Exception.Message)";
$Host.SetShouldExit(1);
}
}

try {
$checkParams = @{ MinimumVersion = $checkParams.RequiredVersion.Split('-')[0] }
$Null = Import-Module PSRule @checkParams -ErrorAction Stop;
$version = (Get-Module PSRule).Version;
}
catch {
Write-Host "::error::An error occurred importing module 'PSRule'. $($_.Exception.Message)";
$Host.SetShouldExit(1);
}

#
# Run assert pipeline
#
Write-Host '';
Write-Host "[info] Using Version: $version";
Write-Host "[info] Using Action: $Env:GITHUB_ACTION";
Write-Host "[info] Using Workspace: $workspacePath";
Write-Host "[info] Using PWD: $PWD";
Write-Host "[info] Using Path: $Path";
Write-Host "[info] Using Source: $Source";
Write-Host "[info] Using Baseline: $Baseline";
Write-Host "[info] Using Conventions: $Conventions";
Write-Host "[info] Using InputType: $InputType";
Write-Host "[info] Using InputPath: $InputPath";
Write-Host "[info] Using Option: $Option";
Write-Host "[info] Using Outcome: $Outcome";
Write-Host "[info] Using OutputFormat: $OutputFormat";
Write-Host "[info] Using OutputPath: $OutputPath";
Write-Host "[info] Using Summary: $Summary";

try {
Push-Location -Path $Path;
$invokeParams = @{
Path = $Source
Style = 'GitHubActions'
ErrorAction = 'Stop'
}
WriteDebug "Preparing command-line:";
WriteDebug ([String]::Concat('-Path ''', $Source, ''''));
if (![String]::IsNullOrEmpty($Baseline)) {
$invokeParams['Baseline'] = $Baseline;
WriteDebug ([String]::Concat('-Baseline ''', $Baseline, ''''));
}
if ($Conventions.Length -gt 0) {
$invokeParams['Convention'] = $Conventions;
WriteDebug ([String]::Concat('-Convention ', [String]::Join(', ', $Conventions)));
}
if (![String]::IsNullOrEmpty($Modules)) {
$moduleNames = $Modules.Split(',', [System.StringSplitOptions]::RemoveEmptyEntries);
$invokeParams['Module'] = $moduleNames;
WriteDebug ([String]::Concat('-Module ', [String]::Join(', ', $moduleNames)));
}
if (![String]::IsNullOrEmpty($Option)) {
$invokeParams['Option'] = $Option;
WriteDebug ([String]::Concat('-Option ', $Option));
}
if (![String]::IsNullOrEmpty($Outcome)) {
$invokeParams['Outcome'] = $Outcome;
WriteDebug ([String]::Concat('-Outcome ', $Outcome));
}
if (![String]::IsNullOrEmpty($OutputFormat) -and ![String]::IsNullOrEmpty($OutputPath) -and $OutputFormat -ne 'None') {
$invokeParams['OutputFormat'] = $OutputFormat;
$invokeParams['OutputPath'] = $OutputPath;
WriteDebug ([String]::Concat('-OutputFormat ', $OutputFormat, ' -OutputPath ''', $OutputPath, ''''));
}
if ($Summary -eq 'true') {
$Env:PSRULE_OUTPUT_JOBSUMMARYPATH = 'reports/ps_rule_summary.md';
$invokeParams['As'] = 'Detail';
}

# repository
if ($InputType -eq 'repository') {
WriteDebug 'Running ''Assert-PSRule'' with repository as input.';
Write-Host '';
Write-Host '---';
Assert-PSRule @invokeParams -InputPath $InputPath -Format File;
}
# inputPath
elseif ($InputType -eq 'inputPath') {
WriteDebug 'Running ''Assert-PSRule'' with input from path.';
Write-Host '';
Write-Host '---';
Assert-PSRule @invokeParams -InputPath $InputPath;
}
}
catch [PSRule.Pipeline.RuleException] {
Write-Host "::error::Rule exception: $($_.Exception.Message)";
Write-Host "$($_.Exception.ScriptStackTrace)";
$Host.SetShouldExit(1);
}
catch [PSRule.Pipeline.FailPipelineException] {
Write-Host "::error::One or more assertions failed. $($_.Exception.Message)";
Write-Host "$($_.Exception.ScriptStackTrace)";
$Host.SetShouldExit(1);
}
catch {
Write-Host "::error::$($_.Exception.Message)";
Write-Host "$($_.Exception.ScriptStackTrace)";
$Host.SetShouldExit(1);
}
finally {
try {
if ($Summary -eq 'true' -and (Test-Path -Path 'reports/ps_rule_summary.md')) {
Get-Content -Path 'reports/ps_rule_summary.md' -Raw > $Env:GITHUB_STEP_SUMMARY;
$Null = Remove-Item -Path 'reports/ps_rule_summary.md' -Force;
}
}
catch {
Write-Host "::warning::Failed to write job summary: $($_.Exception.Message)";
Write-Host "$($_.Exception.ScriptStackTrace)";
}
Pop-Location;
}
Write-Host '---';

0 comments on commit 6e50b7e

Please sign in to comment.