csocketshome-directory

I can't connect to sockets in XDG_RUNTIME_DIR or my own home directory (binding is possible somehow)


this is all happening in a random directory in my home /home/behe/Documents/development/beh/wallpaper_select

client.c

#include <sys/socket.h>
#include <sys/un.h>
#include <string.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
int main() {
  const char* xdg_dir = getenv("XDG_RUNTIME_DIR");
  const char* home = getenv("HOME");
  const char* hello_c = "/hello_c";
  const char* prefix = "/beh/wallpaper_daemon.sock";
  char* S_path = malloc(strlen(xdg_dir)+strlen(prefix)+1);
  //test0
  //strcpy(S_path,"hello_c");
  //works

  //test 1
  //strcpy(S_path,xdg_dir);
  //strcat(S_path,prefix);
  //doesn't work

  //test2
  strcpy(S_path,home);
  strcat(S_path,hello_c);
  //doesn't work

  //test3
  //strcpy(S_path,"/tmp/hello_c");
  //works
  char buf[2400];
  printf("%s\nand its length: %li\ndo I have access: %i\n",S_path,strlen(S_path),access(S_path,F_OK));
  int client_socket = socket(AF_UNIX,SOCK_STREAM,0);
  struct sockaddr_un tosend;
  memset(&tosend,0,sizeof(tosend));
  tosend.sun_family = AF_UNIX;
  strlcpy(tosend.sun_path,S_path,sizeof(tosend.sun_path));
  strlcpy(buf,"this is working finally",sizeof(buf));
  printf("%s\n",tosend.sun_path);
  int connection_status = connect(client_socket,(struct sockaddr *) &tosend,sizeof(struct sockaddr));
  if (connection_status == -1) perror("connection failed: ");
  if (write(client_socket,buf,sizeof(buf)) == -1) perror("write failed: ");
  free(S_path);
  return 0;
}

server.c

#include "sys/types.h"
#include <stdio.h>
#include <sys/socket.h>
#include <string.h> 
#include <sys/un.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/stat.h>
int main() {
  const char* xdg_dir = getenv("XDG_RUNTIME_DIR");
  char* prefix[] = {"/beh","/wallpaper_daemon.sock"};
  const char* home = getenv("HOME");
  const char* hello_c = "/hello_c";
  char* S_path = malloc(strlen(xdg_dir)+strlen(prefix[0])+strlen(prefix[1])+1);
  //test0
  //strcpy(S_path,"hello_c");
  //works

  //test1
  //strcpy(S_path,xdg_dir);
  //strcat(S_path,prefix[0]);
  //printf("%s\n",S_path);
  //mkdir(S_path,0700);
  //strcat(S_path,prefix[1]);
  //doesn't work

  //test2
  strcpy(S_path,home);
  strcat(S_path,hello_c);
  //doesn't work

  //test3
  //strcpy(S_path,"/tmp/hello_c");
  //works
  printf("%s\nand its length: %li\n",S_path,strlen(S_path));
  //get a socket for server
  int S_sock = socket(AF_UNIX,SOCK_STREAM,0);
  if (S_sock == -1) perror("getting socket failed: ");
  struct sockaddr_un S_sock_F;
  //clearing out the ram for the structure
  memset(&S_sock_F,0,sizeof(struct sockaddr_un));
  S_sock_F.sun_family= AF_UNIX;
  strlcpy(S_sock_F.sun_path,S_path,sizeof(S_sock_F.sun_path));
  int S_bind_state = bind(S_sock,
    (struct sockaddr *)&S_sock_F,
    sizeof(S_sock_F));
  if (S_bind_state == -1) perror("binding failed: ");
  int S_listen_state = listen(S_sock,5);
  if (S_listen_state == -1) perror("listening failed: ");
  //int S_accept = accept(S_sock,(struct sockaddr *)&S_sock_F,sizeof(struct sockaddr));
  char buf[2400];
  int S_accept = accept(S_sock,NULL,NULL);
  //waiting for clients
  ssize_t msglen;
  msglen = read(S_accept,buf,sizeof(buf));
  printf("%s\n",buf);
  //perror("");
  close(S_sock);
  unlink(S_sock_F.sun_path);
  free(S_path);
  return 0;
}

the tests are included /tmp works and the said random directory works I didn't try more directories since well why doesn't my home directory work the problem is why can't I create sockets in xdg_runtime_dir or my home dir

steps to reproduce:

step 1: put both files in a random directory in your home

step 2: gcc server.c -o server && ./server

step 3: gcc client.c -o client && ./client

step 4: try out the other tests to see that they work thats all


what value is HOME: /home/myusername what value is xdg_runtime_dir: /run/user/1000 enter image description here

an image of one of the outputs

a better image now (basically can connect to server using nc -U and not with my client so its confirmed the problem is with the client the connection is getting refused from the client) enter image description here I'm using btrfs


Solution

  • I did not reproduce the "No such file or directory" error presented in the question with any variation on the socket path. I did get a "connection refused" error from the client in some cases, however, and it seems to be caused by an error in the client's connect() call:

      int connection_status = connect(client_socket,(struct sockaddr *) &tosend,sizeof(struct sockaddr));
    

    The third argument should be the size of the actual address type, which in this case is struct sockaddr_un, but you have instead passed the size of the struct sockaddr type, which is probably smaller.

    That whether the client works or not depends on the socket path is consistent with that. It is plausible that it works when the offset of the last byte of the path fits within or close to the size of a struct sockaddr, but not when it exceeds that by too much. Changing the above line to ...

      int connection_status = connect(client_socket, (struct sockaddr *) &tosend,
              sizeof(struct sockaddr_un));
    

    ... resolved the issue for me, as did ...

      int connection_status = connect(client_socket, (struct sockaddr *) &tosend,
              sizeof(tosend));
    

    . The latter is probably better form.