Skip to content

Instantly share code, notes, and snippets.

@dreness
Last active December 18, 2019 21:52
Show Gist options
  • Save dreness/dc062abd8d95a6c62f102d3af8d0c4fb to your computer and use it in GitHub Desktop.
Save dreness/dc062abd8d95a6c62f102d3af8d0c4fb to your computer and use it in GitHub Desktop.
#Windows advice

Preamble

I run a fair amount of *nix software on my Windows machine - not because it's easy, but because it has become apparent that this die-hard Mac user needs a more diverse portfolio (for personal reasons; I have successfully avoided Windows for my entire career to date). In fact, it's quite difficult compared to running the same software on macOS, which isn't too surprising since macOS is generally considered to be a member of the *nix family, while Windows is not.

This document contains a random bag of things that will cut you as you attempt to cross the streams, and also the occasional tip or trick.

Condensced wisdom

  • There are many choices of 'environments' for building / running things, such as: MSYS2, Cygwin, Anaconda, VCPKG, WSL. They can be used concurrently (with important caveats; keep reading), and the correct one to use for a given task likely depends on compatability requirements of the software you're making or using.
  • path handling is a mess (WSL purports to ease this pain somewhat)
    • Use ^ instead of " for quoting when possible
  • Embrace the silo
    • Be extremely fastidious, purposeful, and restrained with your environment variable configuration. In general, simply avoid making permanent changes to any environment variables.
    • It's probably best to keep things installed by a given distribution as isolated as possible, unless it's explicitely obvious that sharing is intended.
    • Use the front door. Often times, the environment you're using offers some kind of entry point that prepares the executing process for that environment, by setting (you guessed it) environment variables. You probably want the effects of any environment configuration defined by any of these environments to apply only to the environment it was created for. You can achieve that mostly by using the correct console / shell for each environment, and ONLY that environment. In a nutshell: recycle but DON'T reuse. This also means your shells and consoles (you will have several) are difficult to configure in a consistent way due to the incentive to not re-use. Talk about a first-world problem!

Failure to keep your worlds separated might lead to things like this:

cygwin1.dll has escaped its jurisdiction and now implants itself into the minds of others

MSYS2

Read this twice: https://github.com/msys2/msys2/wiki/MSYS2-introduction:

Key callouts:

Beware that mixing in programs from other MSYS2 installations, Cygwin installations, compiler toolchains or even various other programs is not supported and will probably break things in unexpected ways. Do not have these things in PATH when running MSYS2 unless you know what you're doing.

(you might think you know what you're doing, but allow me to suggest heeding the above warning)

Use msys2 shell for running pacman, makepkg, makepkg-mingw and for building POSIX-dependent software that you don't intend to distribute. Use mingw shells for building native Windows software and other tasks.

From https://github.com/msys2/msys2/wiki/Porting#user-content-filesystem-namespaces:

Paths in mingw Python

The mingw python provided by MSYS2 will try to produce paths that are right for the environment it's running in. According to the value of MSYSTEM, it will use either forward or backward slashes as os.path.sep.

Response files

When passing arguments with absolute paths to native Windows programs (such as /mingw64/bin/gcc), these paths get (in most cases) automatically converted by MSYS2 if they're in POSIX format. However, if you're passing these paths inside a file (e.g. a response file for GCC -- gcc @somefile), they need to be pre-converted to Windows native format.

Command line parsing

Windows programs parse the command line themselves, it isn't parsed for them by the calling process, as on Linux. This means that if wildcards (glob patterns) are to be accepted by the program, it has to be able to expand them somehow. MinGW-w64 supplies the correct start-up code, so it happens automatically, in a manner compatible with MSVC-compiled programs. If undesirable, the behavior can be disabled at program build.

Cygwin/MSYS2 programs have to deal with a mix of both approaches, but they can apparently deal with it. Note that they don't behave exactly like native programs, for example they understand single-quoted arguments.

Unless I'm missing something, I'd say they can only kinda sorta deal with it...

Command-line argument quoting

From "Everyone quotes command line arguments the wrong way":

Because the quoting rules for CommandLineToArgvWcmd's differ from those of cmd, we cannot give a command line intended for the former to the latter and and expect our arguments to survive the trip. Because this difference is subtle, it's easy to forget about it and leave latent bugs in programs that interact with cmd. All of cmd's transformations are triggered by the presence of one of the metacharacters (, ), %, !, ^, ", <, >, &, and |. " is particularly interesting: when cmd is transforming a command line and sees a ", it copies a " to the new command line, then begins copying characters from the old command line to the new one without seeing whether any of these characters is a metacharacter.

"open with" with custom args

  • download and run FileTypesMan
  • select an extension you'd like to customize
  • Choose "new action".
  • Specify a Command something like shown below, being sure to quote the last argument (%1) which passes through the path to the document being opened
"G:\media-autobuild\local64\bin-video\mpv.exe" --config-dir=G:\media-autobuild\local64\etc\mpv --force-window "%1"

FileTypesMan.png

Python

Startup Script

  1. Make a .cmd file:

    set root=C:\Users\andre\work\play-with-mpv
    set py=G:\Apps\Anaconda3\python.exe
    cd %root%
    %py% %root%\play_with_mpv.py
  2. Schedule with Task Scheduler

task-scheduler

pipenv can't find a python version

  File "g:\apps\anaconda3\lib\site-packages\pipenv\core.py", line 360, in find_a_system_python
    python_entry = finder.find_python_version(line)
  File "g:\apps\anaconda3\lib\site-packages\pipenv\vendor\pythonfinder\pythonfinder.py", line 108, in find_python_version
    match = self.windows_finder.find_python_version(
  File "g:\apps\anaconda3\lib\site-packages\pipenv\vendor\pythonfinder\pythonfinder.py", line 63, in windows_finder
    self._windows_finder = WindowsFinder()
  File "<attrs generated init dd945291b0b1e93a8c72035cacb0761f4492555d>", line 13, in __init__
  File "g:\apps\anaconda3\lib\site-packages\pipenv\vendor\pythonfinder\models\windows.py", line 92, in get_versions
    py_version = PythonVersion.from_windows_launcher(version_object)
  File "g:\apps\anaconda3\lib\site-packages\pipenv\vendor\pythonfinder\models\python.py", line 417, in from_windows_launcher
    creation_dict = cls.parse(launcher_entry.info.version)
  File "g:\apps\anaconda3\lib\site-packages\pipenv\vendor\pythonfinder\_vendor\pep514tools\_registry.py", line 75, in __getattr__
    raise AttributeError(attr)
AttributeError: version

This can be caused by stale registry entries pointing to non-existent Python installations. Delete them. See also: pypa/pipenv#3380. You can work around this by specifying the full path to the interpreter when installing new environments:

pipenv install --python=G:\Apps\Anaconda3\python.exe

Building Windows-native code

Sometimes when you're Pythoning, you encounter a module that needs to build native extensions, such as webrtcvad. Even if you have Visual Studio installed, that's probably not enough for everything to 'just work'. You probably want to grab the separate package called Build Tools for Visual Studio 2019. Install the C++ build tools, after which you can launch a "Developer Command Prompt for VS 2019" command prompt, which comes with all the environment variables you should need to build C++ stuff.

WSL

Remember what I said about re-use and silos? I guess not, because you went and configured your wsl user's $HOME to be your Windows user's home. Except hey now, those are two totally different kinds of filesystems. Sure, things appear to work fine for a short while, but one thing that doesn't work as expected from the WSL side is file permissions and ownership, which SSH will be happy to inform you about when you try to use public key auth. For example, note that a wsl resident (in this case Ubuntu) and PowerShell disagree about the permissions on my private key:

wsl view

andre@honk ~ % cat /etc/lsb-release
DISTRIB_ID=Ubuntu
DISTRIB_RELEASE=18.04
DISTRIB_CODENAME=bionic
DISTRIB_DESCRIPTION="Ubuntu 18.04.2 LTS"

andre@honk ~ % ls -al /mnt/c/Users/andre/.ssh/id_rsa
-rwxrwxrwx 1 root root 1675 Dec 30  2017 /mnt/c/Users/andre/.ssh/id_rsa

PowerShell view

PS C:\Users\andre\.ssh> ls id_rsa

    Directory: C:\Users\andre\.ssh


Mode                LastWriteTime         Length Name
----                -------------         ------ ----
-a----       12/30/2017  12:20 AM           1675 id_rsa


PS C:\Users\andre\.ssh> Get-Acl "C:\Users\andre\.ssh\id_rsa"

    Directory: C:\Users\andre\.ssh


Path   Owner      Access
----   -----      ------
id_rsa HONK\andre HONK\andre Allow  FullControl

As of 'Insider Build 17063', a new mount option "-o metadata" has appeared which purports to bridge file ownership and permissions between Windows and wsl. Even though that's almost two years old as I write this, that situation still seems kinda sketch to me, so I abstain. But if you must, read this.

One-liners

  • Disable fast boot, or every other boot might fail (probably specific to my config... Also this is a firmware setting, so why is this even here?!)
  • DO NOT install the Razer Chroma whatever SDK if you enjoy playing any Blizzard game.
  • Don't have too many NVidia SDKs installed concurrently ... or else be prepared for very lengthy driver updates.

Links

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment