我想将我的文件.bashrc
和.bash_login
文件保留在版本控制中,以便可以在所有使用的计算机之间使用它们。问题是我有一些特定于OS的别名,因此我正在寻找一种确定脚本是否在Mac OS X,Linux或Cygwin上运行的方法。
在Bash脚本中检测操作系统的正确方法是什么?
我想将我的文件.bashrc
和.bash_login
文件保留在版本控制中,以便可以在所有使用的计算机之间使用它们。问题是我有一些特定于OS的别名,因此我正在寻找一种确定脚本是否在Mac OS X,Linux或Cygwin上运行的方法。
在Bash脚本中检测操作系统的正确方法是什么?
Answers:
我认为以下应该起作用。我不确定win32
。
if [[ "$OSTYPE" == "linux-gnu"* ]]; then
# ...
elif [[ "$OSTYPE" == "darwin"* ]]; then
# Mac OSX
elif [[ "$OSTYPE" == "cygwin" ]]; then
# POSIX compatibility layer and Linux environment emulation for Windows
elif [[ "$OSTYPE" == "msys" ]]; then
# Lightweight shell and GNU utilities compiled for Windows (part of MinGW)
elif [[ "$OSTYPE" == "win32" ]]; then
# I'm not sure this can happen.
elif [[ "$OSTYPE" == "freebsd"* ]]; then
# ...
else
# Unknown.
fi
msys
Git Bash / msysGit和cygwin
Cygwin
msys
;
。
source somefile ;
,我得到了syntax error near unexpected token elif
。
对于我的.bashrc,我使用以下代码:
platform='unknown'
unamestr=`uname`
if [[ "$unamestr" == 'Linux' ]]; then
platform='linux'
elif [[ "$unamestr" == 'FreeBSD' ]]; then
platform='freebsd'
fi
然后我做类似的事情:
if [[ $platform == 'linux' ]]; then
alias ls='ls --color=auto'
elif [[ $platform == 'freebsd' ]]; then
alias ls='ls -G'
fi
很难看,但是可以用。如果愿意,可以使用case
代替if
。
platform="$(uname | tr '[:upper:]' '[:lower:]')"
吗?然后一定要处理一个案子吗?硬编码对非常特定的输出的依赖项似乎很奇怪。
bash联机帮助页上说,变量OSTYPE存储了操作系统的名称:
OSTYPE
自动设置为描述执行bash的操作系统的字符串。默认值取决于系统。
设置到linux-gnu
这里。
os=${OSTYPE//[0-9.]/}
case $OSTYPE in darwin*) echo I am a Mac ;; esac
if [[ $OSTYPE == darwin* ]]; then echo I am a Mac;fi
$OSTYPE
您可以简单地使用预定义$OSTYPE
变量,例如:
case "$OSTYPE" in
solaris*) echo "SOLARIS" ;;
darwin*) echo "OSX" ;;
linux*) echo "LINUX" ;;
bsd*) echo "BSD" ;;
msys*) echo "WINDOWS" ;;
*) echo "unknown: $OSTYPE" ;;
esac
但是,较旧的外壳(例如Bourne shell)无法识别它。
uname
另一种方法是基于uname
命令检测平台。
请参阅以下脚本(已包含在.bashrc中):
# Detect the platform (similar to $OSTYPE)
OS="`uname`"
case $OS in
'Linux')
OS='Linux'
alias ls='ls --color=auto'
;;
'FreeBSD')
OS='FreeBSD'
alias ls='ls -G'
;;
'WindowsNT')
OS='Windows'
;;
'Darwin')
OS='Mac'
;;
'SunOS')
OS='Solaris'
;;
'AIX') ;;
*) ;;
esac
这是Travis CI上使用的类似版本:
case $(uname | tr '[:upper:]' '[:lower:]') in
linux*)
export TRAVIS_OS_NAME=linux
;;
darwin*)
export TRAVIS_OS_NAME=osx
;;
msys*)
export TRAVIS_OS_NAME=windows
;;
*)
export TRAVIS_OS_NAME=notset
;;
esac
uname
,Oracle掩盖了约定,并在它们的后面放了一个免费版本字符串OSTYPE ==> solaris2.11
。 uname
清洁一点,减少工作量。
检测操作系统和CPU类型也不是那么容易做到的可移植性。我有一个sh
大约100行的脚本,该脚本可在非常广泛的Unix平台上运行:自1988年以来我使用的任何系统。
关键要素是
uname -p
是处理器类型,但通常unknown
在现代Unix平台上。
uname -m
在某些Unix系统上会给出“机器硬件名称”。
/bin/arch
,如果存在,通常会给出处理器的类型。
uname
不带参数的名称将命名为操作系统。
最终,您将不得不考虑平台之间的区别以及您希望如何区分它们。 例如,只是为了让事情变得简单,我把i386
通过i686
,任何“ Pentium*
”任何“ AMD*Athlon*
”所有的x86
。
My ~/.profile
在启动时运行一个脚本,该脚本将一个变量设置为一个字符串,该字符串指示CPU和操作系统的组合。我有特定于平台的bin
,man
,lib
,和include
基于该是那些获得建立目录。然后设置一系列环境变量。因此,例如,可以重新格式化邮件的外壳脚本可以调用,例如,$LIB/mailfmt
它是特定于平台的可执行二进制文件。
如果您想偷工减料,uname -m
Plain uname
会告诉您在许多平台上要了解的内容。在需要时添加其他内容。(并且使用case
,而不是嵌套if
!)
uname -s
呢?
我建议使用此完整的bash代码
lowercase(){
echo "$1" | sed "y/ABCDEFGHIJKLMNOPQRSTUVWXYZ/abcdefghijklmnopqrstuvwxyz/"
}
OS=`lowercase \`uname\``
KERNEL=`uname -r`
MACH=`uname -m`
if [ "{$OS}" == "windowsnt" ]; then
OS=windows
elif [ "{$OS}" == "darwin" ]; then
OS=mac
else
OS=`uname`
if [ "${OS}" = "SunOS" ] ; then
OS=Solaris
ARCH=`uname -p`
OSSTR="${OS} ${REV}(${ARCH} `uname -v`)"
elif [ "${OS}" = "AIX" ] ; then
OSSTR="${OS} `oslevel` (`oslevel -r`)"
elif [ "${OS}" = "Linux" ] ; then
if [ -f /etc/redhat-release ] ; then
DistroBasedOn='RedHat'
DIST=`cat /etc/redhat-release |sed s/\ release.*//`
PSUEDONAME=`cat /etc/redhat-release | sed s/.*\(// | sed s/\)//`
REV=`cat /etc/redhat-release | sed s/.*release\ // | sed s/\ .*//`
elif [ -f /etc/SuSE-release ] ; then
DistroBasedOn='SuSe'
PSUEDONAME=`cat /etc/SuSE-release | tr "\n" ' '| sed s/VERSION.*//`
REV=`cat /etc/SuSE-release | tr "\n" ' ' | sed s/.*=\ //`
elif [ -f /etc/mandrake-release ] ; then
DistroBasedOn='Mandrake'
PSUEDONAME=`cat /etc/mandrake-release | sed s/.*\(// | sed s/\)//`
REV=`cat /etc/mandrake-release | sed s/.*release\ // | sed s/\ .*//`
elif [ -f /etc/debian_version ] ; then
DistroBasedOn='Debian'
DIST=`cat /etc/lsb-release | grep '^DISTRIB_ID' | awk -F= '{ print $2 }'`
PSUEDONAME=`cat /etc/lsb-release | grep '^DISTRIB_CODENAME' | awk -F= '{ print $2 }'`
REV=`cat /etc/lsb-release | grep '^DISTRIB_RELEASE' | awk -F= '{ print $2 }'`
fi
if [ -f /etc/UnitedLinux-release ] ; then
DIST="${DIST}[`cat /etc/UnitedLinux-release | tr "\n" ' ' | sed s/VERSION.*//`]"
fi
OS=`lowercase $OS`
DistroBasedOn=`lowercase $DistroBasedOn`
readonly OS
readonly DIST
readonly DistroBasedOn
readonly PSUEDONAME
readonly REV
readonly KERNEL
readonly MACH
fi
fi
此处有更多示例示例:https : //github.com/coto/server-easy-install/blob/master/lib/core.sh
ls /etc
寻找“ blah-release”(不管是什么),这就是Linux OS-CentOS,Ubuntu等等
在bash中,使用$OSTYPE
和$HOSTTYPE
,如记录所示;这就是我要做的。如果这还不够,并且甚至uname
或uname -a
(或其他适当的选项)没有提供足够的信息,则总是有GNU项目中的config.guess脚本,正是为此目的而编写的。
尝试使用“ uname”。例如,在Linux中:“ uname -a”。
根据手册页,uname符合SVr4和POSIX,因此它也应在Mac OS X和Cygwin上可用,但我无法确认。
顺便说一句:$ OSTYPE也设置为linux-gnu
这里:)
我把这些糖写在我的.bashrc
:
if_os () { [[ $OSTYPE == *$1* ]]; }
if_nix () {
case "$OSTYPE" in
*linux*|*hurd*|*msys*|*cygwin*|*sua*|*interix*) sys="gnu";;
*bsd*|*darwin*) sys="bsd";;
*sunos*|*solaris*|*indiana*|*illumos*|*smartos*) sys="sun";;
esac
[[ "${sys}" == "$1" ]];
}
所以我可以做类似的事情:
if_nix gnu && alias ls='ls --color=auto' && export LS_COLORS="..."
if_nix bsd && export CLICOLORS=on && export LSCOLORS="..."
if_os linux && alias psg="ps -FA | grep" #alternative to pgrep
if_nix bsd && alias psg="ps -alwx | grep -i" #alternative to pgrep
if_os darwin && alias finder="open -R"
if_nix gnu
&都应返回true if_os linux
。
echo $OSTYPE
。为了检测赢10 bash的子系统而言,唯一的解决办法我已经找到了检查“微软”在/proc/sys/kernel/osrelease
uname
要么
uname -a
如果您想了解更多信息
我写了一个个人的Bash库和脚本框架,它使用GNU shtool进行了相当准确的平台检测。
GNU shtool是一组非常可移植的脚本,除其他有用的内容外,还包含“ shtool platform”命令。这是输出:
shtool platform -v -F "%sc (%ac) %st (%at) %sp (%ap)"
在一些不同的机器上:
Mac OS X Leopard:
4.4BSD/Mach3.0 (iX86) Apple Darwin 9.6.0 (i386) Apple Mac OS X 10.5.6 (iX86)
Ubuntu Jaunty server:
LSB (iX86) GNU/Linux 2.9/2.6 (i686) Ubuntu 9.04 (iX86)
Debian Lenny:
LSB (iX86) GNU/Linux 2.7/2.6 (i686) Debian GNU/Linux 5.0 (iX86)
如您所见,这将产生令人满意的结果。GNU shtool有点慢,所以实际上我在脚本调用的系统上的文件中存储和更新平台标识。这是我的框架,因此对我有用,但是您的工作量可能会有所不同。
现在,您必须找到一种将shtool与脚本打包的方法,但这并不是一个艰苦的练习。您也总是可以退回uname输出。
编辑:
我错过了泰迪config.guess
(以某种方式)的帖子。这些是非常相似的脚本,但不相同。我个人也将shtool用作其他用途,并且对我来说效果很好。
您可以使用以下内容:
OS=$(uname -s)
那么您可以在脚本中使用OS变量。
uname -s
和uname
吗有什么不同吗?
下面是一种使用/ etc / lsb-release和/ etc / os-release(取决于您使用的Linux风格)来检测基于Debian和RedHat的Linux OS并基于此采取简单操作的方法。
#!/bin/bash
set -e
YUM_PACKAGE_NAME="python python-devl python-pip openssl-devel"
DEB_PACKAGE_NAME="python2.7 python-dev python-pip libssl-dev"
if cat /etc/*release | grep ^NAME | grep CentOS; then
echo "==============================================="
echo "Installing packages $YUM_PACKAGE_NAME on CentOS"
echo "==============================================="
yum install -y $YUM_PACKAGE_NAME
elif cat /etc/*release | grep ^NAME | grep Red; then
echo "==============================================="
echo "Installing packages $YUM_PACKAGE_NAME on RedHat"
echo "==============================================="
yum install -y $YUM_PACKAGE_NAME
elif cat /etc/*release | grep ^NAME | grep Fedora; then
echo "================================================"
echo "Installing packages $YUM_PACKAGE_NAME on Fedorea"
echo "================================================"
yum install -y $YUM_PACKAGE_NAME
elif cat /etc/*release | grep ^NAME | grep Ubuntu; then
echo "==============================================="
echo "Installing packages $DEB_PACKAGE_NAME on Ubuntu"
echo "==============================================="
apt-get update
apt-get install -y $DEB_PACKAGE_NAME
elif cat /etc/*release | grep ^NAME | grep Debian ; then
echo "==============================================="
echo "Installing packages $DEB_PACKAGE_NAME on Debian"
echo "==============================================="
apt-get update
apt-get install -y $DEB_PACKAGE_NAME
elif cat /etc/*release | grep ^NAME | grep Mint ; then
echo "============================================="
echo "Installing packages $DEB_PACKAGE_NAME on Mint"
echo "============================================="
apt-get update
apt-get install -y $DEB_PACKAGE_NAME
elif cat /etc/*release | grep ^NAME | grep Knoppix ; then
echo "================================================="
echo "Installing packages $DEB_PACKAGE_NAME on Kanoppix"
echo "================================================="
apt-get update
apt-get install -y $DEB_PACKAGE_NAME
else
echo "OS NOT DETECTED, couldn't install package $PACKAGE"
exit 1;
fi
exit 0
输出例如用于Ubuntu Linux操作系统:
delivery@delivery-E5450$ sudo sh detect_os.sh
[sudo] password for delivery:
NAME="Ubuntu"
===============================================
Installing packages python2.7 python-dev python-pip libssl-dev on Ubuntu
===============================================
Ign http://dl.google.com stable InRelease
Get:1 http://dl.google.com stable Release.gpg [916 B]
Get:2 http://dl.google.com stable Release [1.189 B]
...
在所有发行版上都应该可以安全使用。
$ cat /etc/*release
这会产生类似这样的结果。
DISTRIB_ID=LinuxMint
DISTRIB_RELEASE=17
DISTRIB_CODENAME=qiana
DISTRIB_DESCRIPTION="Linux Mint 17 Qiana"
NAME="Ubuntu"
VERSION="14.04.1 LTS, Trusty Tahr"
ID=ubuntu
ID_LIKE=debian
PRETTY_NAME="Ubuntu 14.04.1 LTS"
VERSION_ID="14.04"
HOME_URL="http://www.ubuntu.com/"
SUPPORT_URL="http://help.ubuntu.com/"
BUG_REPORT_URL="http://bugs.launchpad.net/ubuntu/"
根据需要提取/分配变量
注意:在某些设置上。这可能还会给您带来一些您可以忽略的错误。
cat: /etc/upstream-release: Is a directory
[user@host ~]$ cat /etc/*release CentOS release 6.4 (Final) CentOS release 6.4 (Final) CentOS release 6.4 (Final)
cat: /etc/*release: No such file or directory
尝试这个:
DISTRO=$(cat /etc/*-release | grep -w NAME | cut -d= -f2 | tr -d '"')
echo "Determined platform: $DISTRO"
cat: /etc/*-release: No such file or directory
zsh: no matches found: /etc/*release
-我认为这仅适用于linux系统...
您可以使用以下if子句并根据需要展开它:
if [ "${OSTYPE//[0-9.]/}" == "darwin" ]
then
aminute_ago="-v-1M"
elif [ "${OSTYPE//[0-9.]/}" == "linux-gnu" ]
then
aminute_ago="-d \"1 minute ago\""
fi
我在几个Linux发行版中尝试了上述消息,发现以下内容最适合我。这是一个简短,简明的确切单词答案,也适用于Windows上的Bash。
OS=$(cat /etc/*release | grep ^NAME | tr -d 'NAME="') #$ echo $OS # Ubuntu
OS=$( ([[ -f /etc/*release ]] && cat /etc/*release | grep ^NAME | tr -d 'NAME="') || echo "Unknown" )
它将正常工作而不会引发错误。
OS=$( $(compgen -G "/etc/*release" > /dev/null) && cat /etc/*release | grep ^NAME | tr -d 'NAME="') || $( echo "${OSTYPE//[0-9.]/}")
我倾向于将.bashrc和.bash_alias保留在所有平台都可以访问的文件共享上。这就是我在.bash_alias中解决问题的方法:
if [[ -f (name of share)/.bash_alias_$(uname) ]]; then
. (name of share)/.bash_alias_$(uname)
fi
例如,我有一个.bash_alias_Linux:
alias ls='ls --color=auto'
这样我就将平台特定代码和可移植代码分开,您可以对.bashrc执行相同操作
进行以下操作有助于正确检查ubuntu:
if [[ "$OSTYPE" =~ ^linux ]]; then
sudo apt-get install <some-package>
fi