211 lines
4.5 KiB
211 lines
4.5 KiB
<?php |
|
# |
|
# See function sendsockreply($fun, $msg, $tmo) at the end |
|
# |
|
function socktmo($socket, $factor) |
|
{ |
|
# default timeout factor |
|
if ($factor === false) |
|
$factor = 1; |
|
|
|
# on a slower server increase this base value |
|
$tmo = 8; |
|
|
|
$usetmo = $tmo * $factor; |
|
$sec = floor($usetmo); |
|
$usec = floor(($usetmo - $sec) * 1000000); |
|
$tmoval = array('sec' => $sec, 'usec' => $usec); |
|
socket_set_option($socket, SOL_SOCKET, SO_SNDTIMEO, $tmoval); |
|
socket_set_option($socket, SOL_SOCKET, SO_RCVTIMEO, $tmoval); |
|
} |
|
# |
|
# Note that $port in AF_UNIX should be the socket filename |
|
function _getsock($fun, $port, $tmo, $unix=true) |
|
{ |
|
$socket = null; |
|
if ($unix === true) |
|
$socket = socket_create(AF_UNIX, SOCK_STREAM, 0); |
|
else |
|
$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP); |
|
if ($socket === false || $socket === null) |
|
{ |
|
$sle = socket_last_error(); |
|
$sockerr = socket_strerror($sle); |
|
$msg = "$fun() _getsock() create($port) failed"; |
|
error_log("CKPERR: $msg ($sle) '$sockerr'"); |
|
return false; |
|
} |
|
|
|
if ($unix === true) |
|
$res = socket_connect($socket, $port, NULL); |
|
else |
|
$res = socket_connect($socket, '127.0.0.1', $port); |
|
if ($res === false) |
|
{ |
|
// try 3x |
|
if ($unix === true) |
|
$res = socket_connect($socket, $port); |
|
else |
|
{ |
|
sleep(2); |
|
$res = socket_connect($socket, '127.0.0.1', $port); |
|
} |
|
if ($res === false) |
|
{ |
|
if ($unix === true) |
|
$res = socket_connect($socket, $port); |
|
else |
|
{ |
|
sleep(5); |
|
$res = socket_connect($socket, '127.0.0.1', $port); |
|
} |
|
if ($res === false) |
|
{ |
|
$sle = socket_last_error(); |
|
$sockerr = socket_strerror($sle); |
|
if ($unix === true) |
|
$msg = "$fun() _getsock() connect($port) failed 3x"; |
|
else |
|
$msg = "$fun() _getsock() connect($port) failed 3x (+2+5s sleep)"; |
|
error_log("CKPERR: $msg ($sle) '$sockerr'"); |
|
socket_close($socket); |
|
return false; |
|
} |
|
} |
|
} |
|
# Avoid getting locked up for long |
|
socktmo($socket, $tmo); |
|
# Enable timeout |
|
socket_set_block($socket); |
|
return $socket; |
|
} |
|
# |
|
function getsock($fun, $tmo) |
|
{ |
|
return _getsock($fun, '/opt/ckdb/listener', $tmo); |
|
} |
|
# |
|
function readsockline($fun, $socket) |
|
{ |
|
$siz = socket_read($socket, 4, PHP_BINARY_READ); |
|
if ($siz === false) |
|
{ |
|
$sle = socket_last_error(); |
|
$sockerr = socket_strerror($sle); |
|
$msg = "$fun() readsockline() failed"; |
|
error_log("CKPERR: $msg ($sle) '$sockerr'"); |
|
return false; |
|
} |
|
if (strlen($siz) != 4) |
|
{ |
|
$msg = "$fun() readsockline() short 4 read got ".strlen($siz); |
|
error_log("CKPERR: $msg"); |
|
return false; |
|
} |
|
$len = ord($siz[0]) + ord($siz[1])*256 + |
|
ord($siz[2])*65536 + ord($siz[3])*16777216; |
|
$ans = ''; |
|
$left = $len; |
|
while ($left > 0) |
|
{ |
|
$line = socket_read($socket, $left, PHP_BINARY_READ); |
|
if ($line === false) |
|
{ |
|
$sle = socket_last_error(); |
|
$sockerr = socket_strerror($sle); |
|
$msg = "$fun() readsockline() $left failed (len=$len)"; |
|
error_log("CKPERR: $msg ($sle) '$sockerr'"); |
|
return false; |
|
} |
|
$red = strlen($line); |
|
if ($red == 0) |
|
{ |
|
$msg = "$fun() readsockline() incomplete (".($len-$left)." vs $len)"; |
|
$sub = "'".substr($line, 0, 30)."'"; |
|
if (strlen($line) > 30) |
|
$sub .= '...'; |
|
error_log("CKPERR: $msg $sub"); |
|
return false; |
|
} |
|
$left -= $red; |
|
$ans .= $line; |
|
} |
|
return $ans; |
|
} |
|
# |
|
function dosend($fun, $socket, $msg) |
|
{ |
|
$msg .= "\n"; |
|
$len = strlen($msg); |
|
|
|
$sen = $len; |
|
$siz = chr($sen % 256); |
|
$sen = $sen >> 8; |
|
$siz .= chr($sen % 256); |
|
$sen = $sen >> 8; |
|
$siz .= chr($sen % 256); |
|
$sen = $sen >> 8; |
|
$siz .= chr($sen % 256); |
|
|
|
$msg = $siz . $msg; |
|
|
|
$len += 4; |
|
$left = $len; |
|
$ret = false; |
|
while ($left > 0) |
|
{ |
|
$res = socket_write($socket, substr($msg, 0 - $left), $left); |
|
if ($res === false) |
|
{ |
|
$sockerr = socket_strerror(socket_last_error()); |
|
$msg = "$fun() sendsock() failed"; |
|
error_log("CKPERR: $msg '$sockerr'"); |
|
break; |
|
} |
|
if ($res == 0) |
|
{ |
|
$msg = "$fun() sendsock() incomplete (".($len-$left)." vs $len)"; |
|
error_log("CKPERR: $msg"); |
|
break; |
|
} |
|
$left -= $res; |
|
} |
|
if ($left == 0) |
|
$ret = true; |
|
|
|
return $ret; |
|
} |
|
# |
|
function sendsock($fun, $msg, $tmo = false) |
|
{ |
|
$ret = false; |
|
$socket = getsock($fun, $tmo); |
|
if ($socket !== false) |
|
{ |
|
$ret = dosend($fun, $socket, $msg); |
|
socket_close($socket); |
|
} |
|
return $ret; |
|
} |
|
# |
|
# This is the only function in here you call |
|
# You pass it a string $fun for debugging |
|
# and the data $msg to send to ckdb |
|
# and it returns $ret = false on error or $ret = the string reply |
|
# |
|
function sendsockreply($fun, $msg, $tmo = false) |
|
{ |
|
$ret = false; |
|
$socket = getsock($fun, $tmo); |
|
if ($socket !== false) |
|
{ |
|
$ret = dosend($fun, $socket, $msg); |
|
if ($ret !== false) |
|
$ret = readsockline($fun, $socket); |
|
|
|
socket_close($socket); |
|
} |
|
return $ret; |
|
} |
|
# |
|
?>
|
|
|