我有一个shell脚本,在Windows/Cygwin和Mac和Linux上都使用。每个版本都需要稍微不同的变量。

shell/bash脚本如何检测它是在Cygwin、Mac还是Linux上运行?


当前回答

Bash设置shell变量OSTYPE。来自man bash:

自动设置为描述操作系统的字符串 哪个bash正在执行。

这比uname有一个小小的优势,它不需要启动一个新进程,因此执行起来更快。

然而,我无法找到一个权威的期望值列表。对我来说,Ubuntu 14.04设置为“linux-gnu”。我在网上搜到了一些其他的价值。因此:

case "$OSTYPE" in
  linux*)   echo "Linux / WSL" ;;
  darwin*)  echo "Mac OS" ;; 
  win*)     echo "Windows" ;;
  msys*)    echo "MSYS / MinGW / Git Bash" ;;
  cygwin*)  echo "Cygwin" ;;
  bsd*)     echo "BSD" ;;
  solaris*) echo "Solaris" ;;
  *)        echo "unknown: $OSTYPE" ;;
esac

星号在某些情况下很重要——例如,OSX在“darwin”之后附加了一个操作系统版本号。“win”值实际上是“win32”,我被告知-也许有一个“win64”?

也许我们可以一起工作,在这里填充一个验证值的表:

Linux Ubuntu(包括。WSL): Linux -gnu Cygwin 64位:Cygwin Msys/MINGW (Git Bash for Windows): Msys

(如果与现有条目不同,请附加您的值)

其他回答

好的,这是我的方法。

osis()
{
    local n=0
    if [[ "$1" = "-n" ]]; then n=1;shift; fi

    # echo $OS|grep $1 -i >/dev/null
    uname -s |grep -i "$1" >/dev/null

    return $(( $n ^ $? ))
}

e.g.

osis Darwin &&
{
    log_debug Detect mac osx
}
osis Linux &&
{
    log_debug Detect linux
}
osis -n Cygwin &&
{
    log_debug Not Cygwin
}

我在我的dotfiles中使用这个

检测三种不同的操作系统类型(GNU/Linux, Mac OS X, Windows NT)

笔记

在bash脚本中,使用#!/usr/bin/env bash代替#!/bin/sh是为了防止/bin/sh链接到不同平台的不同默认shell所造成的问题,否则会出现意外操作符之类的错误,这就是我的电脑上发生的事情(Ubuntu 64位12.04)。 Mac OS X 10.6.8(雪豹)没有expr程序,除非你安装它,所以我只是使用uname。

设计

使用uname获取系统信息(-s参数)。 使用expr和substr来处理字符串。 使用if elif fi来执行匹配工作。 如果您愿意,可以添加更多的系统支持,只需遵循uname -s规范即可。

实现

#!/usr/bin/env bash

if [ "$(uname)" == "Darwin" ]; then
    # Do something under Mac OS X platform        
elif [ "$(expr substr $(uname -s) 1 5)" == "Linux" ]; then
    # Do something under GNU/Linux platform
elif [ "$(expr substr $(uname -s) 1 10)" == "MINGW32_NT" ]; then
    # Do something under 32 bits Windows NT platform
elif [ "$(expr substr $(uname -s) 1 10)" == "MINGW64_NT" ]; then
    # Do something under 64 bits Windows NT platform
fi

测试

Linux (Ubuntu 12.04 LTS, Kernel 3.2.0)测试OK。 OS X (10.6.8 Snow Leopard)测试正常。 Windows (Windows 7 64位)测试正常。

我学到了什么

检查开头和结尾的引号。 检查是否缺少括号和大括号{}

参考文献

[1] uname - wikipedia [2] shell脚本语法错误:文件意外结束 [3] Bash脚本检测操作系统 [4] BASH编程介绍

我想无名的答案是无敌的,主要是在清洁方面。

虽然执行它需要很长时间,但我发现测试特定文件的存在也会给我更好和更快的结果,因为我没有调用可执行文件:

So,

[-f /usr/bin/cygwin1.dll] && echo是的,Cygwin正在运行

只是使用一个快速的Bash文件存在检查。因为我现在使用的是Windows,所以我无法告诉你任何linux和Mac OS X的具体文件,但我非常确定它们确实存在。: -)

通常,带有各种选项的uname会告诉你正在运行的环境:

pax> uname -a
CYGWIN_NT-5.1 IBM-L3F3936 1.5.25(0.156/4/2) 2008-06-12 19:34 i686 Cygwin

pax> uname -s
CYGWIN_NT-5.1

而且,根据非常有用的肖特(在评论中),uname -s为OSX提供Darwin,为Linux提供Linux,而我的Cygwin提供CYGWIN_NT-5.1。但是你可能要尝试各种不同的版本。

因此,执行此类检查的bash代码将如下所示:

unameOut="$(uname -s)"
case "${unameOut}" in
    Linux*)     machine=Linux;;
    Darwin*)    machine=Mac;;
    CYGWIN*)    machine=Cygwin;;
    MINGW*)     machine=MinGw;;
    *)          machine="UNKNOWN:${unameOut}"
esac
echo ${machine}

注意,我在这里假设您实际上是在CygWin(它的bash shell)中运行,因此路径应该已经正确设置。正如一位评论者指出的那样,你可以从cmd本身运行bash程序,传递脚本,这可能会导致路径没有按需要设置。

如果你这样做,你有责任确保正确的可执行文件(例如CygWin的)被调用,可能是通过事先修改路径或完全指定可执行文件的位置(例如,/c/ CygWin /bin/uname)。

当提出这个问题时,Linux的Windows子系统还不存在。它在我的测试中给出了以下结果:

uname -s -> Linux
uname -o -> GNU/Linux
uname -r -> 4.4.0-17763-Microsoft

这意味着您需要使用uname -r来将其与本地Linux区分开来。