我是新手,期待脚本,所以请原谅我的绊脚石......

下面是我的期望脚本的内容。目的是滚动输出的几个屏幕,在每个屏幕之后都会提示用户“继续?[y/n]”。

最后,当没有更多屏幕时,会显示“%”提示,这应该导致执行退出 while 循环。

    set more_screens 1
    while {$more_screens > 0} {
        sleep 10
        expect {
            "\[y/n]" { send "y\r"}
            "% "  { set more_screens 0 }
        }
     }

相反发生的是......它永远停留在while循环中,一遍又一遍地发送“y”。我已经设置了“exp_internal 1”,并且从那个输出它“似乎”像期望不断重新阅读它已经匹配的文本,因此不断看到“[y/n]”,并在以下情况下不断发送“y” ,实际上只有2屏输出,因此只有2个“继续?[y/n]”提示。

(可能不需要 sleep 语句 - 我只是添加它来解决问题 - 它没有 - 并让我更好地消化调试输出。)

底线......我的代码中有任何明显的错误吗?我会采取任何建议来改进这一点并消除无休止的循环。

在詹姆斯提出了一个有用的建议之后,编辑下面添加到这个问题。

感谢 James 的快速回复和有用的建议!但...

您的方法仍然存在同样的问题(尽管您的方法更加优雅,我会将其添加到我的期望工具包中。)

正如最初指出的那样,问题似乎肯定是每次执行expect 语句都会重新读取已经读取并比较的文本。下面是我执行 James 的“exp_continue”代码时的输出,我设置了“exp_internal 1”以在我的屏幕上获得调试输出......
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
>expect: does "get dump tables\r\n\r% get dump tables\n\r\r\nIfn  TableName           Configured >MaxUse    InUse     LastDropTime\r\n3    cdm_app             100002     190       33        >\r\n3    cdm_conv            2000002    675180    4813      \r\n3    cdm_pdisc           250002     >250002    1304      01-24-2014-19:14:59\r\n3    cdm_kpi             100001     141       25        >\r\n3    cdm_qoe             500003     204918    1578      \r\n3    cdm_qoe_hd          2500003    >582993    1578      \r\n3    cdm_kpi_error_app   100001     5         2         \r\n3    >cdm_kpi_error       100001     7         2         \r\n3    asr_cache           1000000    >1000000   999995    \r\n3    asr_sess            2000000    62670     29748     \r\n3    >asr_conn            3000000    64428     31147     \r\n3    asr_sess_keys       1500000    >1015269   1009049   \r\n3    asr_conn_opts       6000000    0         0         \r\n3    >asr_events          4000000    5239      144       \r\n3    skt_table           2000000    >2000000   2000000   \r\n3    skt_trans           1000000    408020    254674    \r\n3    >ses_sip_db          5000       0         0         \r\n3    ses_gtp_mob_txn     5000       >0         0         \r\nContinue? [y/n]:  " (spawn_id exp6) match glob pattern "[y/n]"? yes
>expect: set expect_out(0,string) "n"
>expect: set expect_out(spawn_id) "exp6"
>expect: set expect_out(buffer) "get dump tables\r\n\r% get dump tables\n\r\r\nIfn"
>send: sending "y\r" to { exp6 }
>expect: continuing expect
>
>
>expect: does "  TableName           Configured MaxUse    InUse     LastDropTime\r\n3    >cdm_app             100002     190       33        \r\n3    cdm_conv            2000002    >675180    4813      \r\n3    cdm_pdisc           250002     250002    1304      01-24-2014->\r\n3    cdm_kpi             100001     141       25        \r\n3    cdm_qoe             500003     >204918    1578      \r\n3    cdm_qoe_hd          2500003    582993    1578      \r\n3    >cdm_kpi_error_app   100001     5         2         \r\n3    cdm_kpi_error       100001     >7         2         \r\n3    asr_cache           1000000    1000000   999995    \r\n3    >asr_sess            2000000    62670     29748     \r\n3    asr_conn            3000000    >64428     31147     \r\n3    asr_sess_keys       1500000    1015269   1009049   \r\n3    >asr_conn_opts       6000000    0         0         \r\n3    asr_events          4000000    >5239      144       \r\n3    skt_table           2000000    2000000   2000000   \r\n3    >skt_trans           1000000    408020    254674    \r\n3    ses_sip_db          5000       >0         0         \r\n3    ses_gtp_mob_txn     5000       0         0         \r\nContinue?
>[y/n]:  " (spawn_id exp6) match glob pattern "[y/n]"? yes
>expect: set expect_out(0,string) "n"
>expect: set expect_out(spawn_id) "exp6"
>expect: set expect_out(buffer) "  TableName           Con"
>send: sending "y\r" to { exp6 }
>^Csighandler: handling signal(2)
>async event handler: Tcl_Eval(exit 130)

 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

在第一次“[y/n]”上的expect匹配之后,它会做一个“expect:continuous expect”(上面文本输出的中间),然后接下来读取的文本块是,除了前几个词,相同的文本块已经阅读和比较。

我错过了什么吗???如果期望语句重新读取已处理的输出,这一定是一个问题,是吗? (我查看了目标系统发送的实际输出,它不会第二次发送相同的文本块。)

同样,我是expect脚本的新手,但我看不到上面调试输出显示的任何其他解释。 (而且,我很抱歉无法正确格式化输出 - 我真的在努力!)

感谢任何有耐心阅读以上所有内容的人,也许有解释或建议。

最佳答案

你需要被 exp_continue 命令解救。 :-)

当遇到该命令时,该命令所做的是留在期望块内并尝试再次匹配,无论可能出现什么新输入。

所以你真的可以将上面的代码缩短为:

expect {
    "\[y/n]" {
        send "y\r"
        exp_continue
    }
    "% " {
        # Do whatever is needed here, after which program flow will continue *outside* of the expect block
    }
}

让我知道这是否适合您!

编辑 - 基于@feenyman99 附加信息:

好的,我明白是什么了。你有错误的模式。通过使用“[y/n]”,将使用单个“n”字符生成匹配。有你匹配的字符串:



expect_out(0,string) 保存匹配的模式。 expect_out(buffer) 保存从缓冲区中移除的输入部分,它保存所有输入直到并包括匹配的模式(从那时起,下一个期望操作将在最后一个匹配模式之后寻找输入的匹配项)。如您所见,它将输入保持到并包括找到的第一个文字 'n' 字符(换行符不计算在内):



所以发生的事情是您的脚本在出现是/否提示之前发送了“y\r”方式。而且,虽然我没有看到其余的日志,但我猜下一场比赛会在点击下一个“n”字符后不久发生。

因此,您需要更改模式匹配语句以匹配是/否提示。最好使它成为正则表达式匹配 (-re)。我测试了以下内容,它有效(在 Tcl 8.4.13 上测试):
expect {
            -re "\\\[y/n]" { send "y\r"}

多个反斜杠是因为反斜杠也是模式匹配器中的转义字符。有点棘手,但有时需要它们。

让我知道这是怎么回事。你现在应该已经准备好了。

PS:这可能会派上用场:http://www.tcl.tk/doc/howto/regexp81.tml

关于while-loop - 期望在 while 循环中 - 永远循环,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/21342588/

10-12 20:19