chore: refactor codebase

- add namespace
- move commands to dedicated file
- collapse linux shells to single process info
This commit is contained in:
Marcello 2023-05-26 16:28:48 +02:00
parent 0f7d7d2ff8
commit c54e96e3e3
5 changed files with 166 additions and 172 deletions

98
src/Commands.cs Normal file
View file

@ -0,0 +1,98 @@
using System.ComponentModel;
using Spectre.Console;
using Spectre.Console.Cli;
namespace ScriptLauncher;
internal sealed class RootCommand : AsyncCommand<RootCommandSettings>
{
private const int Failure = 1;
private const int Success = 0;
public override async Task<int> ExecuteAsync(
CommandContext context,
RootCommandSettings settings
)
{
if (!Directory.Exists(settings.Directory))
{
AnsiConsole.Markup($"[red]The directory '{settings.Directory}' does not exist.[/]");
return Failure;
}
FileInfo[] files;
var finder = new ScriptFinder(settings.Extensions, settings.Directory, settings.Depth);
if (settings.Group)
{
var dict = finder.GetScriptsByDirectory();
if (dict.Count == 0)
{
AnsiConsole.Markup(
$"[red]No scripts script files found in '{finder.RootDirectory}' with extensions '{string.Join(", ", finder.Extensions)}'[/]"
);
return Failure;
}
var dirPrompt = PromptConstructor.GetDirectoryPrompt(dict.Keys.ToArray());
var directoryInfo = AnsiConsole.Prompt(dirPrompt);
files = dict[directoryInfo];
}
else
{
files = finder.GetScripts();
}
if (files.Length == 0)
{
AnsiConsole.Markup(
$"[red]No scripts script files found in '{finder.RootDirectory}' with extensions '{string.Join(", ", finder.Extensions)}'[/]"
);
return Failure;
}
var prompt = PromptConstructor.GetScriptPrompt(files, settings.Brief);
var scripts = AnsiConsole.Prompt(prompt);
try
{
await ScriptExecutor.ExecAsync(scripts, settings.Elevated);
}
catch (Exception ex) when (ex is Win32Exception or InvalidOperationException or PlatformNotSupportedException)
{
AnsiConsole.Markup($"[red]{ex.Message}[/]");
return Failure;
}
return Success;
}
}
internal class RootCommandSettings : CommandSettings
{
[Description("Comma separated list of script extensions")]
[CommandOption("-x|--extensions")]
public string? Extensions { get; init; }
[Description("Search depth")]
[CommandOption("-d|--depth")]
public int Depth { get; init; } = 1;
[Description("Run with elevated privileges")]
[CommandOption("-e|--elevated")]
public bool Elevated { get; init; } = false;
[Description("Group scripts by folder")]
[CommandOption("-g|--group")]
public bool Group { get; init; } = false;
[Description("Show brief information")]
[CommandOption("-b|--brief")]
public bool Brief { get; init; } = false;
[Description("Starting directory (Default: .)")]
[CommandArgument(0, "<path>")]
public string Directory { get; init; } = ".";
}

View file

@ -1,93 +1,6 @@
using System.ComponentModel;
using System.Diagnostics.CodeAnalysis;
using Spectre.Console;
using ScriptLauncher;
using Spectre.Console.Cli;
var app = new CommandApp<RootCommand>();
app.Configure(x => x.SetApplicationName("scrl"));
return app.Run(args);
sealed class RootCommand : AsyncCommand<RootCommandSettings>
{
private const int Failure = 1;
private const int Success = 0;
public override async Task<int> ExecuteAsync(
[NotNull] CommandContext context,
[NotNull] RootCommandSettings settings
)
{
if (!Directory.Exists(settings.Directory))
{
AnsiConsole.Markup($"[red]The directory '{settings.Directory}' does not exist.[/]");
// Environment.ExitCode = 1;
return Failure;
}
FileInfo[] files;
var finder = new ScriptFinder(settings.Extensions, settings.Directory, settings.Depth);
if (settings.Group)
{
var dict = finder.GetScriptsByDirectory();
if (dict.Count == 0)
{
AnsiConsole.Markup(
$"[red]No scripts script files found in '{finder.RootDirectory}' with extensions '{string.Join(", ", finder.Extensions)}'[/]"
);
return Failure;
}
var dirPrompt = PromptConstructor.GetDirectoryPrompt(dict.Keys.ToArray());
var directoryInfo = AnsiConsole.Prompt(dirPrompt);
files = dict[directoryInfo];
}
else
{
files = finder.GetScripts();
}
if (files.Length == 0)
{
AnsiConsole.Markup(
$"[red]No scripts script files found in '{finder.RootDirectory}' with extensions '{string.Join(", ", finder.Extensions)}'[/]"
);
return Failure;
}
var prompt = PromptConstructor.GetScriptPrompt(files, settings.Brief);
var scripts = AnsiConsole.Prompt(prompt);
await ScriptExecutor.ExecAsync(scripts, settings.Elevated);
return Success;
}
}
internal class RootCommandSettings : CommandSettings
{
[Description("Comma separated list of script extensions")]
[CommandOption("-x|--extensions")]
public string? Extensions { get; init; }
[Description("Search depth")]
[CommandOption("-d|--depth")]
public int Depth { get; init; } = 1;
[Description("Run with elevated privileges")]
[CommandOption("-e|--elevated")]
public bool Elevated { get; init; } = false;
[Description("Group scripts by folder")]
[CommandOption("-g|--group")]
public bool Group { get; init; } = false;
[Description("Show brief information")]
[CommandOption("-b|--brief")]
public bool Brief { get; init; } = false;
[Description("Starting directory (Default: .)")]
[CommandArgument(0, "<path>")]
public string Directory { get; init; } = ".";
}

View file

@ -1,11 +1,14 @@
using System.Text;
using Spectre.Console;
static class PromptConstructor
namespace ScriptLauncher;
internal static class PromptConstructor
{
const int ScriptListSize = 15;
private static Style SelectionHighlight => new(decoration: Decoration.Bold | Decoration.Underline);
private static Style SelectionHighlight =>
new(decoration: Decoration.Bold | Decoration.Underline);
private static string FileStyle(FileInfo info, bool brief)
{
@ -31,7 +34,9 @@ static class PromptConstructor
.Title("Select the scripts to execute:")
.NotRequired()
.PageSize(ScriptListSize)
.InstructionsText("[grey](Press [blue]<space>[/] to toggle a script, [green]<enter>[/] to accept)[/]")
.InstructionsText(
"[grey](Press [blue]<space>[/] to toggle a script, [green]<enter>[/] to accept)[/]"
)
.MoreChoicesText("[grey]Move up and down to reveal more options[/]")
.UseConverter(x => FileStyle(x, brief))
.HighlightStyle(SelectionHighlight)

View file

@ -1,33 +1,24 @@
using System.ComponentModel;
using System.Diagnostics;
using Spectre.Console;
static class ScriptExecutor
{
public static async Task ExecAsync(List<FileInfo> files, bool elevated)
namespace ScriptLauncher;
internal static class ScriptExecutor
{
public static async Task ExecAsync(List<FileInfo> files, bool elevated) =>
await Parallel.ForEachAsync(files, (x, ct) => ExecAsync(x, elevated, ct));
}
public static async ValueTask ExecAsync(FileInfo file, bool elevated, CancellationToken cancellationToken = default)
private static async ValueTask ExecAsync(FileInfo file, bool elevated, CancellationToken cancellationToken = default)
{
var process = GetExecutableProcessInfo(file, elevated);
if (process is null) return;
try
if (process is null)
{
return;
}
await (Process.Start(process)?.WaitForExitAsync(cancellationToken) ?? Task.CompletedTask);
}
catch (Exception ex) when (ex is Win32Exception or InvalidOperationException or PlatformNotSupportedException)
{
AnsiConsole.Markup($"[red]{ex.Message}[/]");
}
}
private static ProcessStartInfo? GetExecutableProcessInfo(FileInfo file, bool elevated)
{
return file.Extension switch
private static ProcessStartInfo? GetExecutableProcessInfo(FileInfo file, bool elevated) => file.Extension switch
{
".bat" or ".cmd" => new ProcessStartInfo
{
@ -43,28 +34,13 @@ static class ScriptExecutor
Verb = elevated ? "runas /user:Administrator" : string.Empty,
WorkingDirectory = file.DirectoryName
},
".sh" => new ProcessStartInfo
".sh" or ".zsh" or ".fish" => new ProcessStartInfo
{
FileName = "bash",
FileName = "sh",
Arguments = $"-c ./{file.Name}",
Verb = elevated ? "sudo" : string.Empty,
WorkingDirectory = file.DirectoryName
},
".zsh" => new ProcessStartInfo
{
FileName = "zsh",
Arguments = $"-c ./{file.Name}",
Verb = elevated ? "sudo" : string.Empty,
WorkingDirectory = file.DirectoryName
},
".fish" => new ProcessStartInfo
{
FileName = "fish",
Arguments = $"-c ./{file.Name}",
Verb = elevated ? "sudo" : string.Empty,
WorkingDirectory = file.DirectoryName
},
_ => null
var _ => null
};
}
}

View file

@ -1,6 +1,6 @@
using Spectre.Console;
namespace ScriptLauncher;
readonly struct ScriptFinder
internal readonly struct ScriptFinder
{
static readonly string[] DefaultExtensions = new[] { ".ps1", ".*sh", ".bat", ".cmd" };
public string[] Extensions { get; }
@ -10,7 +10,9 @@ readonly struct ScriptFinder
public ScriptFinder(string? extensions, string directory, int depth)
{
Extensions = extensions?.Split(new[] { ',', ' ' }, StringSplitOptions.RemoveEmptyEntries)
Extensions =
extensions
?.Split(new[] { ',', ' ' }, StringSplitOptions.RemoveEmptyEntries)
.ToHashSet()
.Select(x => $".{x.TrimStart('.')}")
.ToArray() ?? DefaultExtensions;