Sunday, September 13, 2020

PHP Unix Socket under Windows

It's been a while that Windows 10 have support of Unix Domain Socket/UDS (although with limitation but that doesn't matter for this blog). While PHP language has some questionable design decisions, let me tell you something.

PHP already prepared for AF_UNIX support in Windows older than you think.

Checking the PHP source code, the earliest presence of sockaddr_un struct in the code existed since PHP 4.1.0. First let's test whetever UDS is supported in PHP 7.4. The server code

<?php
$sock = socket_create(AF_UNIX, SOCK_STREAM, 0);
$path = __DIR__ . DIRECTORY_SEPARATOR . 'uds.sock';
if (file_exists($path))
unlink($path);
if (!socket_bind($sock, $path))
{
echo "Failed to bind Unix Socket", PHP_EOL;
exit(1);
}
if (!socket_listen($sock, 5))
{
echo "Failed to listen", PHP_EOL;
exit(1);
}
$response = "Hello from PHP " . PHP_VERSION . " from " . PHP_OS;
while (true)
{
$s = socket_accept($sock);
$data = "HTTP/1.0 200 OK\r\nContent-Type: text/plain\r\nDate: " . gmdate('D, d M Y H:i:s T') . "\r\nContent-Length: " . strlen($response) . "\r\n\r\n$response";
echo "Received connection", PHP_EOL;
socket_send($s, $data, strlen($data), 0);
socket_close($s);
}
view raw uds_server.php hosted with ❤ by GitHub

In my machine, that code runs (no error message)


So, the client code:

<?php
$sock = socket_create(AF_UNIX, SOCK_STREAM, 0);
$path = __DIR__ . DIRECTORY_SEPARATOR . 'uds.sock';
if (socket_connect($sock, $path))
{
$buf = "";
if (socket_recv($sock, $buf, 1024, MSG_WAITALL))
echo "Received: $buf", PHP_EOL;
socket_close($sock);
}
else
echo "Failed to connect", PHP_EOL;
view raw uds_client.php hosted with ❤ by GitHub

Running the client code prints the following message.

That means UDS is working. "Wait, that's probably PHP completely emulating UDS in Windows", fine, let's write the client code with C instead!

/* clang uds_client.c -lws2_32 -Wno-deprecated-declarations */
#include <errno.h>
#include <stdio.h>
#include <string.h>
#include <winsock2.h>
#include <windows.h>
struct sockaddr_un
{
ADDRESS_FAMILY sun_family;
char sun_path[108];
};
int main()
{
WSADATA wsaData = {0};
struct sockaddr_un sockIn = {AF_UNIX};
SOCKET sock;
char buf[1024];
int err = 0, recvSize;
if (WSAStartup(MAKEWORD(2,2), &wsaData) != 0)
{
err = WSAGetLastError();
puts("FATAL CANNOT INIT SOCKET!");
goto exit;
}
sock = socket(AF_UNIX, SOCK_STREAM, 0);
memcpy(sockIn.sun_path, "uds.sock", sizeof("uds.sock"));
printf("Path: %s\n", sockIn.sun_path);
if (connect(sock, (struct sockaddr *) &sockIn, sizeof(sockIn.sun_path)) == -1)
{
err = WSAGetLastError();
puts("Connect error");
goto exit;
}
recvSize = recv(sock, buf, 1024, 0);
if (recvSize <= 0)
{
err = WSAGetLastError();
puts("Recv error");
goto exit;
}
printf("Received: %s\n", buf);
exit:
if (err)
printf("WSA Error (%d)\n", err);
WSACleanup();
return err > 0;
}
view raw uds_client.c hosted with ❤ by GitHub

Compiling that code and running it shows this output.


That also means PHP didn't emulate UDS and uses WinSock directly.

So, UDS works in PHP 7.4.1 under Windows. What about earlier version? It should be, right? Since reference to UDS in Windows existed since PHP 4.1.0. So let's try with PHP 5.3.19.

Note: The reason I choose PHP 5.3.19 because I didn't look at older commits that reference sockaddr_un in Windows, and I just see that it's there since 4.1.0 while writing this blog, after doing all testing.



The reverse is also possible, i.e. PHP 7.4.1 connecting to UDS which was created for PHP 5.3.19 and vice versa, but that will clutter this blog post with images.

This is surprising discovery in my opinion. What if PHP team already predicted this blog post after all? Well ...

That's story for another blog post.