如何使用 Tcl 的(interp)bgerror

发布于 2024-08-04 06:05:07 字数 1021 浏览 6 评论 0原文

我正在尝试 在从属解释器中运行 tclhttpd 但稍作修改以便在 tclkit 中运行。下面的代码“运行”(我可以点击 http://localhost:8015 ),但永远不会到达 put 行底部因为“服务器没有返回,它进入[vwait Forever]”。但是当我尝试“after 0 技巧”时,例如在“$httpd eval $cmd”行前面添加“after 0”,服务器根本不运行,所以我认为“错误必须由 bgerror 处理”

但是我找不到如何使用 bgerror 的好例子,而且我的研究表明,现在的惯例是使用“interp bgerror”。请参阅 http://www2.tcl.tk/_ 返回的前几个示例/gsearch?S=bgerror;第一个链接包含措辞“填写使用 bgerror 的有用技巧和示例”,但没有我可以辨别如何应用的示例,第二个链接的结论是“我对如何使用它的示例感兴趣”。

package require starkit
starkit::startup

set httpd_args [list]
set httpd [interp create]
$httpd eval "set argc [llength $httpd_args]"
set cmdargv "set argv [list $httpd_args ]"
$httpd eval "set topdir $starkit::topdir"
$httpd eval $cmdargv

set cmd [list source [file join $starkit::topdir bin/httpd.tcl]]
$httpd eval $cmd

puts "if seeing this controlled has returned"

I'm trying to run tclhttpd in a slave interpreter but slightly modified so as to run within a tclkit. The code below "runs" (I can hit http://localhost:8015) but never reaches the puts line at the bottom because "the server does not return, it enters [vwait forever]". But when I try "the after 0 trick", e.g. prepending "after 0 " to the line "$httpd eval $cmd", the server does not run at all, so I presume "errors have to be handled by bgerror"

However I cannot find good examples of how to use bgerror, plus my research shows that now the convention is to use "interp bgerror". Please see the first couple of examples returned by http://www2.tcl.tk/_/gsearch?S=bgerror; the first link contains the verbiage "fill in useful tricks and examples for using bgerror" but then there are no samples I can discern how to apply, and the second link concludes "I am interested in examples how this is supposed to be used."

package require starkit
starkit::startup

set httpd_args [list]
set httpd [interp create]
$httpd eval "set argc [llength $httpd_args]"
set cmdargv "set argv [list $httpd_args ]"
$httpd eval "set topdir $starkit::topdir"
$httpd eval $cmdargv

set cmd [list source [file join $starkit::topdir bin/httpd.tcl]]
$httpd eval $cmd

puts "if seeing this controlled has returned"

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(3

梨涡 2024-08-11 06:05:07

我不太明白你问的问题。听起来你的目标是在一个解释器中启动一个 http 服务器,但以某种方式与主解释器交互。是这样吗?如果是这样,那与 bgerror 有什么关系?

您是否知道,即使您在单独的解释器中运行服务器,它也不是在单独的线程中运行?也就是说,当任一解释器被 vwait 阻止时,您无法 (*) 与主解释器交互。

(*) 你可以,如果你的交互采用 Tk 小部件的形式,并且也利用事件循环

至于如何使用 bgerror,有几种工作方式。默认机制调用函数“bgerror”,您可以定义该函数来执行您想要的任何操作。它需要一个字符串(错误消息的文本)并对其执行某些操作。该某事可能是将错误打印到 stdout、在对话框中显示、将其写入文件等。

作为示例,请考虑此交互式会话:

% proc bgerror {s} {puts "hey! I caught an error: $s"}
% # after 30 seconds, throw an error
% after 30000 {error "this is an error"}
after#0
% # after 40 seconds, terminate the event loop
% after 40000 {set ::done 1}
after#1
% # start the event loop
% vwait ::done
hey! I caught an error: this is an error
% # this prompt appears after 40 seconds or so

您还可以注册自己的错误处理程序,如“interp bgerror”文档中所述。出现在 tcl 8.5 中,尽管它有一个 bug 直到 8.5.3 才修复。

例如:

% set foo [interp create]
interp0
% $foo eval {proc myErrorHandler {args} {puts "myErrorHandler: $args"}}
% $foo bgerror myErrorHandler
myErrorHandler
% # after 30 seconds, throw an error
% $foo eval {after 30000 {error "this is an error"}}
after#0
% # after 40 seconds, terminate the loop
% $foo eval {after 40000 {set ::done 1}}
after#1
% $foo eval {vwait ::done}
myErrorHandler: {this is an error} {-code 1 -level 0 -errorcode NONE -errorinfo {this is an error
    while executing
"error "this is an error""
    ("after" script)} -errorline 1}
% # this prompt appears after 40 seconds or so

这有助于回答您的问题吗?

I don't quite understand the question you are asking. It sounds like your goal is to start up an http server in one interpreter but somehow interact with the main interpreter. Is that right? If so, what does that have to do with bgerror?

Are you aware that even though you are running the server in a separate interpreter, it is not running in a separate thread? That is, you can't (*) interact with the main interpreter while either interpreter is blocked by a vwait.

(*) you can, if your interaction takes the form of Tk widgets that also take advantage of the event loop

As for how to use bgerror, There are a couple of ways that it works. The default mechanism calls the function 'bgerror" which you may define to do whatever you want. It takes a single string (the text of an error message) and does something with it. That something could be to print the error to stdout, show it in a dialog, write it to a file, etc.

As an example, consider this interactive session:

% proc bgerror {s} {puts "hey! I caught an error: $s"}
% # after 30 seconds, throw an error
% after 30000 {error "this is an error"}
after#0
% # after 40 seconds, terminate the event loop
% after 40000 {set ::done 1}
after#1
% # start the event loop
% vwait ::done
hey! I caught an error: this is an error
% # this prompt appears after 40 seconds or so

You can also register your own error handler, as described in the documentation for "interp bgerror". This came along in tcl 8.5, though it had a bug that wasn't fixed until 8.5.3.

For example:

% set foo [interp create]
interp0
% $foo eval {proc myErrorHandler {args} {puts "myErrorHandler: $args"}}
% $foo bgerror myErrorHandler
myErrorHandler
% # after 30 seconds, throw an error
% $foo eval {after 30000 {error "this is an error"}}
after#0
% # after 40 seconds, terminate the loop
% $foo eval {after 40000 {set ::done 1}}
after#1
% $foo eval {vwait ::done}
myErrorHandler: {this is an error} {-code 1 -level 0 -errorcode NONE -errorinfo {this is an error
    while executing
"error "this is an error""
    ("after" script)} -errorline 1}
% # this prompt appears after 40 seconds or so

Does this help answer your question?

总以为 2024-08-11 06:05:07

根据OP的评论完全编辑...

after 0技巧是以下行:

after 0 $httpd eval $cmd

它的作用是告诉interp将有问题的命令($http eval $cmd)添加到事件队列,这意味着它将运行一次事件循环已启动(如果已启动则返回)。您可以在该页面的以下评论中看到对事件循环的依赖(作者:Jacob Levy):

我应该注意,这取决于事件循环是否处于活动状态。

我的猜测是,您正在运行普通的 Tclsh,这意味着您永远不会进入事件循环(Wish shell 在脚本末尾进入事件循环,而 Tcl shell 不会)。进入事件循环的标准方法是在到达 Tcl 代码末尾时运行以下命令:

# Enter the event loop and stay in it until someone 
# sets the "forever" variable to something
vwait forever

也就是说,vwait 之后的任何内容都不会运行,直到退出事件循环之后。如果您希望 httpd 与您的代码并行运行,您需要:

  • 使用多个线程,或者编写您的...这实际上并不是
  • 基于事件的硬代码...这需要您了解基于事件的编程足以防止代码片段缺乏执行时间。

希望有帮助。

Completely edited based on the OP's comments...

The after 0 trick is the following line:

after 0 $httpd eval $cmd

What this does is tell the interp to add the command in question ($http eval $cmd) to the event queue, which means it will run once the event loop is started (or returned to if it's already started). You can see the reliance on the event loop in the following comment from that page (by Jacob Levy):

I should note that this depends on the event loop being active.

My guess is that you're running a plain Tclsh, which means you never enter the event loop (the Wish shell enters the event loop at the end of the script, the Tcl shell does not). The standard way to enter the event loop is to run the following command once you get to the end of your Tcl code:

# Enter the event loop and stay in it until someone 
# sets the "forever" variable to something
vwait forever

That being said, anything you have after the vwait will not run until after the event loop is exited. If you want the httpd to run in parallel to your code, you need to either:

  • Use multiple threads, or write your ... which really isn't that hard
  • code to be event based ... which requires you understand even based programming well enough to prevent pieces of code from being starved of execution time.

Hope that helps.

可爱暴击 2024-08-11 06:05:07

如果我正确理解了你想要做什么,你的代码应该类似于:

set httpd_id [thread::create -preserved]
thread::send $http_id "source [file join $starkit::topdir bin/httpd.tcl]"

这样你就可以让 TclHttpd 在线程中运行,而不用担心 vwait 问题

如果你还想在执行过程中收到有关任何错误的通知httpd 执行时,TclHttp 将所有错误发送到日志文件。您可以配置日志的路径:

Log_SetFile "/logs/httpd_log"

您需要有 httpd::log 包。

我希望这有帮助。

If I've understood correctly what you want to do, your code should look similar to that:

set httpd_id [thread::create -preserved]
thread::send $http_id "source [file join $starkit::topdir bin/httpd.tcl]"

In this way you'll have TclHttpd running in a thread, without worrying for the vwait problem

If you also want to be informed about any error during the httpd execution, TclHttp sends all the errors to a log file. You can configure the path of the Log doing:

Log_SetFile "/logs/httpd_log"

You need to have the httpd::log package.

I hope this helps.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文