macOS Sierra中的更改阻止了“说”在PHP脚本中执行
作者:互联网
我有一个带有一个小站点的macOS服务器,它使用say命令将文本片段转换为音频.
随着升级到Sierra,一切顺利,除了一件事:当我的PHP脚本中的exec()包装时,say命令不再起作用.
该页面刚刚超时.也没有发现任何错误.
<?php
try {
exec('/usr/bin/say "hello"');
}
catch (Exception $e) { echo $e->getMessage(); }
?>
通常我会使用say -o filename保存音频片段,但我尝试了所有变体以及其他正常工作的shell命令,包括在我的输出文件夹中创建文件.
有趣的是,如果我从命令行运行它,它可以工作 – 要么大声说,要么创建一个输出文件.
macOS Sierra有PHP 5.6.24所以我认为safe_mode不适用,righht?
我想强调的是,使用新操作系统,PHP或say命令的更改是最新的.是的我确实调查并尝试了不同的输出和stderr重定向,但脚本只是挂起.
在Activity Viewer中查看say命令(相当于顶层的GUI),我尝试对其进行采样,不确定它是否有帮助:
2695 Thread_1742595 DispatchQueue_1: com.apple.main-thread (serial)
+ 2695 start (in libdyld.dylib) + 1 [0x7fffb0f58255]
+ 2695 ??? (in say) load address 0x10907d000 + 0x1fac [0x10907efac]
+ 2695 NewSpeechChannel (in SpeechSynthesis) + 52 [0x7fff9acd3f19]
+ 2695 SpeechChannelHandle::SpeechChannelHandle() (in SpeechSynthesis) + 265 [0x7fff9acd797f]
+ 2695 dispatch_once_f (in libdispatch.dylib) + 38 [0x7fffb0f220e5]
+ 2695 _dispatch_client_callout (in libdispatch.dylib) + 8 [0x7fffb0f22128]
+ 2695 ___ZN13SpeechGlobals8InstanceEv_block_invoke (in SpeechSynthesis) + 28 [0x7fff9acd54da]
+ 2695 SpeechGlobals::SpeechGlobals() (in SpeechSynthesis) + 471 [0x7fff9acd56db]
+ 2695 xpc_connection_send_message_with_reply_sync (in libxpc.dylib) + 154 [0x7fffb11b65a8]
+ 2695 dispatch_mach_send_with_result_and_wait_for_reply (in libdispatch.dylib) + 45 [0x7fffb0f3cf39]
+ 2695 _dispatch_mach_send_and_wait_for_reply (in libdispatch.dylib) + 591 [0x7fffb0f3cad4]
+ 2695 mach_msg (in libsystem_kernel.dylib) + 55 [0x7fffb107e867]
+ 2695 mach_msg_trap (in libsystem_kernel.dylib) + 10 [0x7fffb107f41a]
2695 Thread_1742600
2695 start_wqthread (in libsystem_pthread.dylib) + 13 [0x7fffb116f211]
2695 _pthread_wqthread (in libsystem_pthread.dylib) + 1426 [0x7fffb116f7b5]
2695 __workq_kernreturn (in libsystem_kernel.dylib) + 10 [0x7fffb10874e6]
从打开的文件和端口,我可以看到我将stdout和stderr都设置为/ private / var / log / apache2 / error_log,但根本没有任何显示.
此外,试图通过更精细的运行捕获输出,但没有快乐,只是超时(脚本文件夹也是可写的):
<?php
try {
$pipes = array();
proc_close(proc_open("say hi", array(0 => array("pipe", "r"), 1 => array("pipe", "r"), 2 => array("pipe", "r")), $pipes, dirname(__FILE__), null));
} catch (Exception $e) { error_log($e->getMessage()); }
?>
更新:High Sierra是一样的.
最终更新:安装Mojave后,删除了大部分Server.app功能,我添加了MAMP来处理这个任务.如果你愿意,可以自己听听 – 这是在macspeaks.com.
解决方法:
我也经历过同样的问题.
这是我刚刚提出的解决方法,但老实说,我认为这是一个很大的麻烦只是为了能够从基于apache的php脚本播放音频.我对于为什么会发生这种情况有一些想法,但经过多次测试,我似乎破坏了我自己的理论.我认为这可能与没有活跃的TTY有关.我无法通过启动带有sudo -i的shell来播放音频,而且还有许多来自php的其他功能,但我能够使用来自本地终端的所有相同命令播放音频,事实证明,也是通过SSH进入电脑,导致我最新的解决方案.再一次,我认为这是过度杀伤,但到目前为止,我能够在我的基于Web的PHP驱动脚本(主要是地理围栏相关的)中获得音频的唯一方法.
所以,我们走了,是的,我理解这涉及的风险和愚蠢:
在我的php脚本中,我使用与此类似的命令在/ tmp目录中生成一个音频文件:
exec "sudo -u <username> /usr/bin/say -o /tmp/outputfile.aiff --voice=Ava \"<What to Say>\"";
然后在生成音频文件之后,我发现从apache / php中播放它(并实际听到输出)的唯一方法是使用expect脚本在本地ssh并播放它.所以我的下一行是:
exec "sudo -u <username> -i ~<username>/expectscript";
我期望的脚本如下:
#!/usr/bin/expect -f
spawn /usr/bin/ssh localhost
expect "Password"
send "<PASSWORD>\r"
expect "<username>"
send "/usr/bin/afplay /tmp/outputfile.aiff\r"
expect "<username>"
send "/usr/bin/touch /tmp/touchthis\r"
expect "<username>"
send "exit\r"
确保替换所有<用户名>上面有你的用户名(显然没有<>)和< PASSWORD>用你的密码.如果您的bash提示不包含您的用户名,您可能需要调整expect脚本,因为这是我在expect脚本中用来查找返回的提示.触摸只是为了确定期望脚本是否有效,并且您可以参考上次触摸文件的时间.
我希望这开启了关于实际导致这种情况的讨论,我们可以确定一个更合理的解决方案.我找了很多兔子试图寻找不同的方法来实现这一点,我创建了Automator应用程序并从PHP调用了它们(没有用.)我在shell中作为我的用户启动了shell,所有命令的执行总是成功完成,没有音频输出.我尝试过的所有解决方案都可以在终端(甚至是终端的php)中正常工作,而不是来自Apache / PHP.
标签:php,exec,macos-sierra 来源: https://codeday.me/bug/20190711/1429461.html