For a comprehensive overview of how PowerShell interacts with external programs, which includes sending data to them, see this answer.
When PowerShell interprets output from external programs (such as ytd
in your case), it assumes that the output uses the character encoding reflected in [Console]::OutputEncoding
.
Note:
Interpreting refers to cases where PowerShell captures (e.g., $output = ...
), relays (e.g., ... | Select-String ...
), or redirects (e.g., ... > output.txt
) the external program's output.
By contrast, printing directly to the display may not be affected, because PowerShell then isn't involved, and certain CLIs adjust their behavior when their stdout isn't redirected to print directly to the console with full Unicode support (which explains why the characters looked as expected in your console when ytd
's output printed directly to it).
If the encoding reported by [Console]::OutputEncoding
is not the same encoding used by the external program at hand, PowerShell misinterprets the output.
To fix that, you must (temporarily) set [Console]::OutputEncoding]
to match the encoding used by the external program.
For instance, let's assume an executable foo.exe
that outputs UTF-8-encoded text:
# Save the current encoding and switch to UTF-8.
$prev = [Console]::OutputEncoding
[Console]::OutputEncoding = [System.Text.UTF8Encoding]::new()
# PowerShell now interprets foo's output correctly as UTF-8-encoded.
# and $output will correctly contain CJK characters.
$output = foo https://example.org -e
# Restore the previous encoding.
[Console]::OutputEncoding = $prev
Important:
With the specific program at hand, youtube-dl
, js2010 has discovered that capturing in a variable works without extra effort if you pass --encoding utf-16
.
The reason this works is that the resulting UTF16-LE-encoded output is preceded by a BOM (Byte-Order Mark).
(Note that --encoding utf-8
does not work, because youtube-dl
then does not emit a BOM.)
Windows PowerShell is capable of detecting and properly decoding UTF-16LE-encoded and UTF-8-encoded text irrespective of the effective [Console]::OutputEncoding]
IF AND ONLY IF the output is preceded by a BOM.
Caveats:
This does not work in PowerShell Core (v6+, on any of the supported platforms).
Even in Windows PowerShell you'll rarely be able to take advantage of this obscure behavior, because using a BOM in stdout output is atypical (it is typically only used in files).
ytd
here refers toyoutube-dl
, which is an open source program and is not a virus. It shouldn't be confused with the other program called "YTD Downloader", which is an ad-filled garbage. – Kirwin