]> git.saurik.com Git - redis.git/commitdiff
Support for Redis to listen on a Unix socket
authorPieter Noordhuis <pcnoordhuis@gmail.com>
Sun, 1 Aug 2010 20:55:24 +0000 (22:55 +0200)
committerPieter Noordhuis <pcnoordhuis@gmail.com>
Sun, 1 Aug 2010 20:55:24 +0000 (22:55 +0200)
src/anet.c
src/anet.h
src/redis-benchmark.c
src/redis.c

index 4fe811a117ee0a8b290b04e1ef1e3df273ec9fde..514da8face645d2d4e2edac9b478cee2a0e0febc 100644 (file)
@@ -32,6 +32,7 @@
 
 #include <sys/types.h>
 #include <sys/socket.h>
+#include <sys/un.h>
 #include <netinet/in.h>
 #include <netinet/tcp.h>
 #include <arpa/inet.h>
@@ -177,6 +178,43 @@ int anetTcpNonBlockConnect(char *err, char *addr, int port)
     return anetTcpGenericConnect(err,addr,port,ANET_CONNECT_NONBLOCK);
 }
 
+int anetUnixGenericConnect(char *err, char *path, int flags)
+{
+    int s;
+    struct sockaddr_un sa;
+
+    if ((s = socket(AF_LOCAL,SOCK_STREAM,0)) == -1) {
+        anetSetError(err, "creating socket: %s\n", strerror(errno));
+        return ANET_ERR;
+    }
+    sa.sun_family = AF_LOCAL;
+    strncpy(sa.sun_path,path,sizeof(sa.sun_path)-1);
+    if (flags & ANET_CONNECT_NONBLOCK) {
+        if (anetNonBlock(err,s) != ANET_OK)
+            return ANET_ERR;
+    }
+    if (connect(s,(struct sockaddr*)&sa,sizeof(sa)) == -1) {
+        if (errno == EINPROGRESS &&
+            flags & ANET_CONNECT_NONBLOCK)
+            return s;
+
+        anetSetError(err, "connect: %s\n", strerror(errno));
+        close(s);
+        return ANET_ERR;
+    }
+    return s;
+}
+
+int anetUnixConnect(char *err, char *path)
+{
+    return anetUnixGenericConnect(err,path,ANET_CONNECT_NONE);
+}
+
+int anetUnixNonBlockConnect(char *err, char *path)
+{
+    return anetUnixGenericConnect(err,path,ANET_CONNECT_NONBLOCK);
+}
+
 /* Like read(2) but make sure 'count' is read before to return
  * (unless error or EOF condition is encountered) */
 int anetRead(int fd, char *buf, int count)
@@ -245,6 +283,31 @@ int anetTcpServer(char *err, int port, char *bindaddr)
     return s;
 }
 
+int anetUnixServer(char *err, char *path)
+{
+    int s;
+    struct sockaddr_un sa;
+
+    if ((s = socket(AF_LOCAL,SOCK_STREAM,0)) == -1) {
+        anetSetError(err, "socket: %s\n", strerror(errno));
+        return ANET_ERR;
+    }
+    memset(&sa,0,sizeof(sa));
+    sa.sun_family = AF_LOCAL;
+    strncpy(sa.sun_path,path,sizeof(sa.sun_path)-1);
+    if (bind(s,(struct sockaddr*)&sa,SUN_LEN(&sa)) == -1) {
+        anetSetError(err, "bind: %s\n", strerror(errno));
+        close(s);
+        return ANET_ERR;
+    }
+    if (listen(s, 511) == -1) { /* the magic 511 constant is from nginx */
+        anetSetError(err, "listen: %s\n", strerror(errno));
+        close(s);
+        return ANET_ERR;
+    }
+    return s;
+}
+
 int anetAccept(char *err, int serversock, char *ip, int *port)
 {
     int fd;
index ce0f4778799824c8669e7487035dab6db614c057..10db3d2fc9d64bbd5cf2956f29e8f9bef9a8881a 100644 (file)
 
 int anetTcpConnect(char *err, char *addr, int port);
 int anetTcpNonBlockConnect(char *err, char *addr, int port);
+int anetUnixConnect(char *err, char *path);
+int anetUnixNonBlockConnect(char *err, char *path);
 int anetRead(int fd, char *buf, int count);
 int anetResolve(char *err, char *host, char *ipbuf);
 int anetTcpServer(char *err, int port, char *bindaddr);
+int anetUnixServer(char *err, char *path);
 int anetAccept(char *err, int serversock, char *ip, int *port);
 int anetWrite(int fd, char *buf, int count);
 int anetNonBlock(char *err, int fd);
index 123d81180e76a76dca76fe298d81ff66f34dfe1a..b3e729f128a7e811982adb57727f8c5a1f68f775 100644 (file)
@@ -71,6 +71,7 @@ static struct config {
     aeEventLoop *el;
     char *hostip;
     int hostport;
+    char *hostsocket;
     int keepalive;
     long long start;
     long long totlatency;
@@ -340,7 +341,11 @@ static client createClient(void) {
     client c = zmalloc(sizeof(struct _client));
     char err[ANET_ERR_LEN];
 
-    c->fd = anetTcpNonBlockConnect(err,config.hostip,config.hostport);
+    if (config.hostsocket == NULL)
+        c->fd = anetTcpNonBlockConnect(err,config.hostip,config.hostport);
+    else
+        c->fd = anetUnixNonBlockConnect(err,config.hostsocket);
+
     if (c->fd == ANET_ERR) {
         zfree(c);
         fprintf(stderr,"Connect: %s\n",err);
@@ -436,6 +441,9 @@ void parseOptions(int argc, char **argv) {
         } else if (!strcmp(argv[i],"-p") && !lastarg) {
             config.hostport = atoi(argv[i+1]);
             i++;
+        } else if (!strcmp(argv[i],"-s") && !lastarg) {
+            config.hostsocket = argv[i+1];
+            i++;
         } else if (!strcmp(argv[i],"-d") && !lastarg) {
             config.datasize = atoi(argv[i+1]);
             i++;
@@ -459,7 +467,8 @@ void parseOptions(int argc, char **argv) {
             printf("Wrong option '%s' or option argument missing\n\n",argv[i]);
             printf("Usage: redis-benchmark [-h <host>] [-p <port>] [-c <clients>] [-n <requests]> [-k <boolean>]\n\n");
             printf(" -h <hostname>      Server hostname (default 127.0.0.1)\n");
-            printf(" -p <hostname>      Server port (default 6379)\n");
+            printf(" -p <port>          Server port (default 6379)\n");
+            printf(" -s <socket>        Server socket (overrides host and port)\n");
             printf(" -c <clients>       Number of parallel connections (default 50)\n");
             printf(" -n <requests>      Total number of requests (default 10000)\n");
             printf(" -d <size>          Data size of SET/GET value in bytes (default 2)\n");
@@ -505,6 +514,7 @@ int main(int argc, char **argv) {
 
     config.hostip = "127.0.0.1";
     config.hostport = 6379;
+    config.hostsocket = NULL;
 
     parseOptions(argc,argv);
 
index c8b1c78146bd66d4b9bd4ef2f5c9807f2a6fa691..397dd65a98db11f2f1bd94717c27ad98ec150cb1 100644 (file)
@@ -773,9 +773,16 @@ void initServer() {
     createSharedObjects();
     server.el = aeCreateEventLoop();
     server.db = zmalloc(sizeof(redisDb)*server.dbnum);
-    server.fd = anetTcpServer(server.neterr, server.port, server.bindaddr);
+    if (server.bindaddr == NULL || inet_aton(server.bindaddr,NULL)) {
+        /* Either no address given, or it can be correctly parsed. */
+        server.fd = anetTcpServer(server.neterr, server.port, server.bindaddr);
+    } else {
+        /* Bind to a socket */
+        unlink(server.bindaddr); /* don't care if this fails */
+        server.fd = anetUnixServer(server.neterr,server.bindaddr);
+    }
     if (server.fd == -1) {
-        redisLog(REDIS_WARNING, "Opening TCP port: %s", server.neterr);
+        redisLog(REDIS_WARNING, "Opening port/socket: %s", server.neterr);
         exit(1);
     }
     for (j = 0; j < server.dbnum; j++) {