3

环境:Apache/2.2.11 (Win32) mod_apreq2-20051231/2.6.2-dev mod_perl/2.0.4-dev Perl/v5.10.0

情况与此讨论列表帖子中描述的情况非常相似,除了在 win32 上。

我在 httpd.conf 中有这个:

PerlModule Apache2::Reload
PerlInitHandler Apache2::Reload
PerlSetVar ReloadAll Off
PerlSetVar ReloadModules "MyPackage::*"

...以及只有 mod-perl 脚本处理。

我有一个使用 MyPackage 模块的脚本,它正在工作。

我破坏了模块,然后重新加载脚本。这个错误很有帮助,告诉我在哪里破坏了模块。

(如果此时我再次重新加载,它只告诉我“未定义的子例程 &ModPerl::ROOT::ModPerl::Registry...”,因为它第一次无法加载文件。但无论哪种方式如下行为仍然发生。)

我恢复了中断,并触摸了脚本文件,这样它就会重新加载模块,然后重新加载。现在它说:

Attempt to reload MyPackage/Foo.pm aborted.
Compilation failed in require at E:/dev/test.pl line 4.

即使我触摸了脚本和模块,我也无法正确地重新加载它,除非重新启动 Web 服务器。

仅破坏脚本本身(而不是模块)可以正常工作:适当的错误并将其改回会导致它在重新加载时再次工作。

在完成这些事情之后,我在测试之前重新启动了 Web 服务器:

  1. 我尝试进行跟踪,但它继续出错的行是 ModPerl/RegistryCooker.pm 第 204 行,这只是 eval{} 整个脚本的行。

  2. 我尝试在脚本和模块中将“使用警告 FATAL => 'all'”更改为“使用警告”。没有什么不同。

  3. 我试过禁用我的自定义 $SIG{__DIE__} 功能。没有什么不同。(好吧,当然,只是在出现错误的地方,但产生的错误是一样的。)

  4. 根据开头的讨论链接,发现 MaxRequestsPerChild 一直为 0,我尝试了 ThreadsPerChild 1,但没有区别。我尝试将 MaxRequestsPerChild 设置为 1,这解决了这个问题的奇怪行为,但在每次请求后重新启动 Web 服务器:

    Child 7072: Process exiting because it reached MaxRequestsPerChild. Signaling the parent to restart a new child process.
    Parent: Received restart signal -- Restarting the server.
    

    这不是一个好的解决方案,因为我有大量代码在第一次点击页面时运行。

  5. 同样根据讨论,我将 httpd 作为服务运行,所以我在服务参数窗口中添加了 -X 并点击开始,它仍然试图在整整三分钟后启动(通常在 3 秒内启动。)甚至超时信息。通过任务管理器终止进程并验证我无法从 Web 浏览器访问该页面。从命令行启动 httpd -X。仍然与此问题顶部的行为相同。我还发现运行 httpd -? 时没有列出 -X 很奇怪。也许它在 win32 MPM 上不可用?

  6. 在该线程中,大卫指出:

    我对此类问题进行故障排除的经验表明,被卸载的包可能删除了存储在重新加载的模块的包空间中的值(可能在 BEGIN 块时间设置),随后的要求没有恢复。

    但这不适用于我的代码。我介绍的“破坏脚本”错误只是在已经存在的行之上添加了一个额外的“我的 $var”行,这样第二个就会抱怨它已经被声明了。

在每次重新加载后(无论是自动,通过 MaxRequestsPerChild,还是像以前一样手动)都没有重新启动 Web 服务器,就没有办法处理 mod_perl2 模块吗?

4

1 回答 1

2

将 sub 复制到您从中调用它的脚本中,并将此代码放在副本之前:

package MyPackage::Foo;
no warnings 'redefine';

完成更改后,将 sub 移回实际模块。

于 2009-02-17T20:25:48.400 回答