AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / computer / 问题 / 1873582
Accepted
mjbatty
mjbatty
Asked: 2025-01-28 23:13:41 +0800 CST2025-01-28 23:13:41 +0800 CST 2025-01-28 23:13:41 +0800 CST

测试脚本是否在另一个脚本内运行

  • 772

[更新]感谢大家迄今为止的评论和想法。基于这些,我更新并扩展了原始问题,添加了更多信息和示例。

我们假设目前没有实际用例,唯一的目标是了解这是否可能,如果可能,如何实现。

我想运行一个脚本并确定它是直接由用户运行还是在另一个脚本中调用。我还想检查它是否有来源。

这意味着有 4 个主要测试用例:

  • 在终端运行,无需来源
  • 在终端运行,来源
  • 在脚本中运行(不在终端中),没有来源
  • 在脚本中运行(不在终端中),来源

我的大脑已经非常生疏,谷歌和 ChatGPT 为我提供了一系列检查,以测试它们是否在终端中运行,但没有一个能够在所有 4 个调用中给出正确的结果。

测试已完成check-term并且有一个调用脚本check-term-driver,均在下面。

如果我按照上面给出的顺序运行 4 个测试用例,对于给定的正确测试号(假设测试号为 n),我希望输出为:

n: in terminal:     not sourced: ...
n: in terminal:     sourced: ...
n: not in terminal: not sourced: ...
n: not in terminal: sourced: ...

在所有 4 个调用中,对是否存在来源的测试都是正确的。

检查项脚本:

#!/usr/bin/env bash

IS_TERMINAL=true
NOT_TERMINAL=false
TEST_NUM=0
SOURCED_TEXT="undefined"

print_result() {
  TEST_NUM=$((TEST_NUM + 1))
  if [ "$1" = "$IS_TERMINAL" ]; then
    printf "%2d: %-16s %-12s %s\n" "$TEST_NUM" "in terminal:" "$SOURCED_TEXT:" "$2"
  else
    printf "%2d: %-16s %-12s %s\n" "$TEST_NUM" "not in terminal:" "$SOURCED_TEXT:" "$2"
  fi
}

# First check if script is sourced or not.
if [[ "${BASH_SOURCE[0]}" == "$0" ]]; then
  SOURCED_TEXT="not sourced"
else
  SOURCED_TEXT="sourced"
fi

# Tests run individually in expanded if/else for clarity.
# - test condition described by the result text.
if [ -t 0 ]; then
    print_result "$IS_TERMINAL" "stdin is a terminal"
else
    print_result "$NOT_TERMINAL" "stdin is not a terminal"
fi

if [ -t 1 ]; then
    print_result "$IS_TERMINAL" "stdout is a terminal"
else
    print_result "$NOT_TERMINAL" "stdout is not a terminal"
fi

if [ -t 2 ]; then
    print_result "$IS_TERMINAL" "stderr is a terminal"
else
    print_result "$NOT_TERMINAL" "stderr is not a terminal"
fi

if [[ "$-" == *i* ]]; then
    print_result "$IS_TERMINAL" "interactive shell"
else
    print_result "$NOT_TERMINAL" "non-interactive shell"
fi

if [ -n "$(tty)" ]; then
    print_result "$IS_TERMINAL" "has a controlling terminal"
else
    print_result "$NOT_TERMINAL" "does not have a controlling terminal"
fi

if [ -p /dev/stdin ]; then
    print_result "$NOT_TERMINAL" "running in a pipeline"
else
    print_result "$IS_TERMINAL" "not running in a pipeline"
fi

if [ -z "$(jobs -p)" ]; then
    print_result "$IS_TERMINAL" "running in foreground"
else
    print_result "$NOT_TERMINAL" "running in background"
fi

# Get ID of process that started the script.
PPID_VALUE=$(ps -o ppid= -p $$ | awk 'NR==2 {print $1}')

# If ID can't be found, give it a default value.
[ -z "$PPID_VALUE" ] && PPID_VALUE=1

# Check if attached or not.
if [ "$PPID_VALUE" -ne 1 ]; then
    print_result "$IS_TERMINAL" "attached to a parent process"
else
    print_result "$NOT_TERMINAL" "detached from a parent process"
fi

# Check if the parent process is a shell
PARENT_CMD=$(ps -o args= -p "$PPID_VALUE" 2>/dev/null)
if echo "$PARENT_CMD" | grep -qE '/(bash|zsh|sh)(\s|$)'; then
    print_result "$IS_TERMINAL" "parent process is a shell"
else
    print_result "$NOT_TERMINAL" "parent process is not a shell"
fi

驱动脚本:

#!/usr/bin/env bash

./check-term
echo ""
. ./check-term
bash
  • 2 2 个回答
  • 184 Views

2 个回答

  • Voted
  1. Cyrus
    2025-01-30T02:41:19+08:002025-01-30T02:41:19+08:00

    使用 bash 版本 3.2.57、4.4.18、5.1.16 和 5.2 进行测试:

    #!/bin/bash
    
    if [[ ${#BASH_LINENO[@]} -gt 1 ]]; then
      echo "sourced, script"
    else
      if [[ "$0" = "${BASH_SOURCE[0]}" ]]; then
        
        parent=$(ps -o comm= $PPID)
        if [[ "$parent" =~ "bash" ]]; then
          echo "not sourced, terminal"
        else
          echo "not sourced, script ($parent)"
        fi
    
      else
        if [[ "${BASH_REMATCH[0]}" != "bash" ]]; then
        echo "sourced, terminal"
        fi
      fi
    fi
    

    bash注 1:如果二进制文件已重命名且其名称不再包含,则此代码将无法正常工作bash。

    注 2:如果从函数执行此代码,则该代码不起作用。

    • 1
  2. Best Answer
    mjbatty
    2025-01-30T16:53:03+08:002025-01-30T16:53:03+08:00

    实现此目的最可靠的方法(到目前为止)是使用 pai-to 建议的环境变量;这适用于所有场景,包括嵌套函数调用。

    Cyrus 建议的解决方案是独立的(没有外部依赖),如果我确信测试不会在函数中使用,我会使用它。

    虽然环境变量解决方案有效,但我仍然更喜欢没有外部依赖的解决方案;如果有人有解决方案,请告诉我。

    这是一个使用环境变量的工作演示。

    检查期限:

    #!/usr/bin/env bash
    
    IS_SOURCED=
    IN_SCRIPT=
    
    # Check if not sourced.
    if [[ "${BASH_SOURCE[0]}" == "$0" ]]; then
      IS_SOURCED=false
    else
      IS_SOURCED=true
    fi
    
    # Check if called by script.
    if [[ -n "$CT_CALLER" ]]; then
      IN_SCRIPT=true
    else
      IN_SCRIPT=false
    fi
    
    if [ "$IS_SOURCED" = false ] && [ "$IN_SCRIPT" = false ]; then
        echo "in terminal: not sourced"
    elif [ "$IS_SOURCED" = true ] && [ "$IN_SCRIPT" = false ]; then
        echo "in terminal: sourced"
    elif [ "$IS_SOURCED" = false ] && [ "$IN_SCRIPT" = true ]; then
        echo "in script: not sourced"
    elif [ "$IS_SOURCED" = true ] && [ "$IN_SCRIPT" = true ]; then
        echo "in script: sourced"
    else
        echo "Error: Invalid state"
    fi
    

    司机:

    #!/usr/bin/env bash
    
    # Set calling script name"
    export CT_CALLER="ct-driver"
    
    ./ct-sea
    . ./ct-sea
    
    • 1

相关问题

  • 在非 root 用户上用 bash 替换 zsh

  • 在 macOS High Sierra 的终端中设置环境变量时遇到问题

  • 对于 cp 或 mv,是否有等同于 cd - 的东西?

  • Notify-发送窗口下出现的通知

  • 如何从 WSL 打开 office 文件

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    如何减少“vmmem”进程的消耗?

    • 11 个回答
  • Marko Smith

    从 Microsoft Stream 下载视频

    • 4 个回答
  • Marko Smith

    Google Chrome DevTools 无法解析 SourceMap:chrome-extension

    • 6 个回答
  • Marko Smith

    Windows 照片查看器因为内存不足而无法运行?

    • 5 个回答
  • Marko Smith

    支持结束后如何激活 WindowsXP?

    • 6 个回答
  • Marko Smith

    远程桌面间歇性冻结

    • 7 个回答
  • Marko Smith

    子网掩码 /32 是什么意思?

    • 6 个回答
  • Marko Smith

    鼠标指针在 Windows 中按下的箭头键上移动?

    • 1 个回答
  • Marko Smith

    VirtualBox 无法以 VERR_NEM_VM_CREATE_FAILED 启动

    • 8 个回答
  • Marko Smith

    应用程序不会出现在 MacBook 的摄像头和麦克风隐私设置中

    • 5 个回答
  • Martin Hope
    Vickel Firefox 不再允许粘贴到 WhatsApp 网页中? 2023-08-18 05:04:35 +0800 CST
  • Martin Hope
    Saaru Lindestøkke 为什么使用 Python 的 tar 库时 tar.xz 文件比 macOS tar 小 15 倍? 2021-03-14 09:37:48 +0800 CST
  • Martin Hope
    CiaranWelsh 如何减少“vmmem”进程的消耗? 2020-06-10 02:06:58 +0800 CST
  • Martin Hope
    Jim Windows 10 搜索未加载,显示空白窗口 2020-02-06 03:28:26 +0800 CST
  • Martin Hope
    andre_ss6 远程桌面间歇性冻结 2019-09-11 12:56:40 +0800 CST
  • Martin Hope
    Riley Carney 为什么在 URL 后面加一个点会删除登录信息? 2019-08-06 10:59:24 +0800 CST
  • Martin Hope
    zdimension 鼠标指针在 Windows 中按下的箭头键上移动? 2019-08-04 06:39:57 +0800 CST
  • Martin Hope
    jonsca 我所有的 Firefox 附加组件突然被禁用了,我该如何重新启用它们? 2019-05-04 17:58:52 +0800 CST
  • Martin Hope
    MCK 是否可以使用文本创建二维码? 2019-04-02 06:32:14 +0800 CST
  • Martin Hope
    SoniEx2 更改 git init 默认分支名称 2019-04-01 06:16:56 +0800 CST

热门标签

windows-10 linux windows microsoft-excel networking ubuntu worksheet-function bash command-line hard-drive

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve