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

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


当前回答

检测三种不同的操作系统类型(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编程介绍

其他回答

好的,这是我的方法。

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中使用这个

# This script fragment emits Cygwin rulez under bash/cygwin
if [[ $(uname -s) == CYGWIN* ]];then
    echo Cygwin rulez
else 
    echo Unix is king
fi

如果uname -s命令的前6个字符为“CYGWIN”,则假设系统为CYGWIN

http://en.wikipedia.org/wiki/Uname

所有你需要的信息。谷歌是你的朋友。

使用uname -s命令查询系统名称。

麦克:达尔文 Cygwin: CYGWIN_… Linux:多种多样,Linux为大多数

通常,带有各种选项的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)。

根据Albert的回答,我喜欢使用$COMSPEC来检测Windows:

#!/bin/bash

if [ "$(uname)" == "Darwin" ]
then
 echo Do something under Mac OS X platform
elif [ "$(expr substr $(uname -s) 1 5)" == "Linux" ]
then
  echo Do something under Linux platform
elif [ -n "$COMSPEC" -a -x "$COMSPEC" ]
then 
  echo $0: this script does not support Windows \:\(
fi

这避免了为$OS解析Windows名称的变体,也避免了解析uname的变体,如MINGW、Cygwin等。

背景:%COMSPEC%是一个Windows环境变量,指定命令处理器(即Windows shell)的完整路径。这个变量的值通常是%SystemRoot%\system32\cmd.exe,它通常计算为C:\Windows\system32\cmd.exe。