Php sockets,Apache和CentOS 6.3出现"Permission denied"错误

4
我要做的是运行一个简单的PHP脚本,检查游戏服务器是否在线并获取一些信息。我在本地使用WAMP Server运行完全相同的脚本,在其中取消注释php_openssl.dll和php_sockets.dll,然后 - voila - 它按预期工作。
但是接下来是我们的生产环境!我通常使用Debian工作,但是我们的主机决定在我们的专用服务器上安装CentOS,因为NIC在Debian中失败了,自那以后一直麻烦不断。
我克服了一些问题,现在面临这个问题:如何修复PHP sockets?我阅读到需要php-common,所以我用以下命令安装了它:
# yum install php-common

然后我检查了phpinfo(),得到了以下结果

'./configure'  '--with-openssl' '--enable-sockets' ...

如果你问我,这看起来很不错,openssl和sockets都已安装并应该正常工作,但事实并非如此。我在Stack Overflow上找到了这个脚本:

<?php
//just in case
if (!extension_loaded('sockets')) {
die('The sockets extension is not loaded.');
}
echo '<p><strong>Establishing connection...</strong></p>';
$socket = socket_create(AF_INET,SOCK_STREAM,0);
if (!socket_connect($socket, "stackoverflow.com", 80))
{
die('Socket error : '.socket_strerror(socket_last_error()));
}

echo '<p><strong>Connection successful!</strong></p>';

$request = join("\n",array(
"GET / HTTP/1.1",
"Connection: close",
"Host: stackoverflow.com",
"User-Agent: Mozilla/5.0 (Windows NT 6.1)",
"Accept: text/html,*/*;q=0.8",
""));
socket_write($socket,$request,strlen($request));

$response = socket_read($socket,2048);


echo "<p><strong>This is the received data : </strong></p>";
echo '<pre>'.htmlentities($response).'</pre>';

socket_close($socket);

这个输出完全让我感到惊讶:
Establishing connection...

Connection successful!

This is the received data :

HTTP/1.0 408 Request Time-out
Cache-Control: no-cache
Connection: close
Content-Type: text/html

<html><body><h1>408 Request Time-out</h1>
Your browser didn't send a complete request in time.
</body></html>

但是,我的脚本仍然无法正常工作?:(我已经通过禁用iptables来尝试解决

# service iptables stop

只是为了确保这不是防火墙问题。以下是发生的情况,当我打开页面时,游戏服务器名称没有显示,而是出现以下内容:

警告:socket_connect():无法连接[13]:权限被拒绝,在/var/www/test.php第9行<<<

这是我正在使用的代码,它是来自xPaw的开源代码,所以我不会得到任何荣誉:

<?php
function queryserver( $IP, $Port = 25565, $Timeout = 2 )
{
$Socket = Socket_Create( AF_INET, SOCK_STREAM, SOL_TCP );
Socket_Set_Option( $Socket, SOL_SOCKET, SO_SNDTIMEO, array( 'sec' => (int)$Timeout, 'usec' => 0 ) );
Socket_Set_Option( $Socket, SOL_SOCKET, SO_RCVTIMEO, array( 'sec' => (int)$Timeout, 'usec' => 0 ) );
if( $Socket === FALSE || Socket_Connect( $Socket, $IP, (int)$Port ) === FALSE )
{
return FALSE;
}
Socket_Send( $Socket, "\xFE\x01", 2, 0 );
$Len = @Socket_Recv( $Socket, $Data, 256, 0 );
Socket_Close( $Socket );
if( $Len < 4 || $Data[ 0 ] !== "\xFF" )
{
return FALSE;
}
$Data = SubStr( $Data, 3 );
$Data = iconv( 'UTF-16BE', 'UTF-8', $Data );

if( $Data[ 1 ] === "\xA7" && $Data[ 2 ] === "\x31" )
{
$Data = Explode( "\x00", $Data );
return Array(
'HostName'   => $Data[ 3 ],
'Players'    => IntVal( $Data[ 4 ] ),
'MaxPlayers' => IntVal( $Data[ 5 ] ),
'Protocol'   => IntVal( $Data[ 1 ] ),
'Version'    => $Data[ 2 ]
);
}
$Data = Explode( "\xA7", $Data );
return Array(
'HostName'   => SubStr( $Data[ 0 ], 0, -1 ),
'Players'    => isset( $Data[ 1 ] ) ? IntVal( $Data[ 1 ] ) : 0,
'MaxPlayers' => isset( $Data[ 2 ] ) ? IntVal( $Data[ 2 ] ) : 0,
'Protocol'   => 0,
'Version'    => '1.3'
);
}

$test = queryserver('SERVERADDRESSHERE'); // (i tried multiple, which are all working @wamp)
$echo = $test['HostName'] . "<<< <br />";
echo $echo;

为什么要使用套接字来实现这个? - VuesomeDev
3个回答

25

以 root 用户身份在 Linux 系统终端中输入此命令。

setsebool -P httpd_can_network_connect 1

拯救我的一天!谢谢。 - KaKa
哇,你救了我的一天。 - Atif

0

(代表原帖作者发布)。

我找到了答案。看起来SELinux正在阻止套接字连接。使用“#setenforce Permissive”禁用SE Linux,现在它可以正常工作了。以防万一有人遇到相同的问题。


0
我曾经遇到过类似的问题,所以我想在这里留下一个答案,为下一个遇到这个问题的人提供帮助。一种解决方法是,在不禁用/限制SELinux的情况下,改变你的httpd周围的策略。
希望这能为下一个阅读此文的人节省两天时间。哈哈
chcon -R -t httpd_sys_content_t /var/www
chcon -R -t httpd_sys_content_rw_t /var/www/html

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接