PowerShell
PowerShell is a hilariously verbose command environment, both in terms of every damn command and also in terms of what it spits out at you when something is wrong---at least, at the times where it doesn't just refuse to say anything at all. Unhelpfully verbose or unhelpfully taciturn seem to be its only two modes! However, PowerShell is necessary in many scenarios thanks to being more featureful and modern than Windows' vestigial cmd.exe environment.
So here are some important things, presuming you already know how to use normal shells on #Linux or such. And if you don't, stop reading this and learn to be comfortable there first; it's a much less user-hostile place to be, far fewer lines of red error text suddenly thrust upon you. Seriously, it's generally far less of a hassle even on Windows to just use the Windows Subsystem for Linux! (You can even pipe to it, ex dir | wsl grep -i SomeFilenameIAmLookingFor.)
Running normal commands with quoted command or argument paths
When using quoted arguments or even just commands, PowerShell goes "oh hey strings" because fucking everything's an object in PowerShell and so suddenly it's thinking you're trying to do string operations when you're just trying to run a damn command with a space in it or something.
The trick is to prepend & to the start of the command. Ex.
& "C:\path\to\someplace some asshole put space\something.exe" "D:\Silly Path\"
Basic Equivalents
Posix-y command | PowerShell equivalent | Documentation | Notes |
---|---|---|---|
cat | Get-Content | ss64, Microsoft | |
grep | Select-String -Pattern | ss64, Microsoft | |
head / tail | it's complicated | see below | |
ls | ls / dir / Get-ChildItem | ss64, Microsoft | Acts like ls -l, no easy equiv of just ls |
pushd | pushd / Push-Location | ss64 | Works in DOS too, less annoying than cd there in fact since it'll traverse filesystem boundaries |
tail -f | Get-Content -Wait | Easily chokes on bad characters and if it does so it actually won't wait, so in practice often just doesn't work | |
top | see below | see below | see below |
tree | tree /F | Microsoft | tree looks fine on screen, but attempts to grep or FindStr fill the output with ????s. By default only prints folders, print files too with /F. This is a basic EXE, it is NOT an alias for a PowerShell equivalent. |
unzip | Expand-Archive | ss64, Microsoft | |
wget | Invoke-WebRequest | ss64, Microsoft | In fact running wget will already get you Invoke-WebRequest, which saves the content to an object in memory rather than defaulting to file output like wget does, making this a surprising alias. You'll need to run, say, wget url://etc -OutFile filename.exe. |
Alternatives and complexities
head and tail
You can use Get-Content -Last 10 / Get-Content -First 10 (documentation: ss64, Microsoft) but only for text files.
Things in PowerShell tend to be objects, which is cool I guess but makes everything more awkward too. So for example for ls -ltr | tail you'll have to do something like Get-ChildItem '..\path\' | Sort-Object -Property LastWriteTime | Select-Object -Last 10.
top
You almost just have to grab some equivalent compiled for Windows (for example bottom aka btm is a nice cross-platform atop-like program). Install with winget install Clement.bottom.
The best you can really do natively in PowerShell is crazy one-liners like for example:
While(1) { $p = get-counter '\Process(*)\% Processor Time'; cls; $p.CounterSamples | sort -des CookedValue | select -f 15 | ft -a}
less and cat
For a nicer equivalent of the less and cat commands, bat aka batcat is the way to go. winget install sharkdp.bat.
Environment variables
Access normal environment variables with $env:VARIABLENAME, ie. instead of %USERPROFILE% for Windows' equivalent of $HOME it's $env:USERPROFILE. Contain in curly braces to avoid ambiguity in parsing, ex. ${env:USERPROFILE}.
Examples
# Get the current state of scheduled tasks with the string "T401" in their name schtasks | Select-String -Pattern "T401" # See if anything with "notepad" in the executable name are running tasklist | Select-String -Pattern "notepad" # Get all the version info on all the EXEs in the local directory Get-ChildItem "*.exe" | % {$_.VersionInfo} | Select * # Get only some info Get-ChildItem "*.exe" | % {$_.VersionInfo} | Select OriginalFilename,FileVersionRaw
External links for other problems
- Last Author
- keithzg
- Last Edited
- Mar 8 2024, 4:51 PM