C и Python - общение с сокетами - PullRequest
7 голосов
/ 25 июля 2011

Я пытаюсь использовать доменные сокеты UNIX для связи между программой на C и скриптом Python. Скрипт Python отправляет данные через доменные сокеты UNIX в программу на языке C.

Вот соответствующий код из моей программы на C:

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/un.h>
#include <sys/types.h>
#include <sys/socket.h>    
#define UNIX_PATH_MAX 100

int main(void)
{
struct sockaddr_un address;
int socket_fd, connection_fd;
socklen_t address_length;
pid_t child;

socket_fd = socket(AF_UNIX, SOCK_STREAM, 0);
if (socket_fd < 0){
    printf("socket() failed\n");
    return 1;
} 

unlink("/tmp/demo_socket");
memset(&address, 0, sizeof(struct sockaddr_un));

address.sun_family = AF_UNIX;
snprintf(address.sun_path, UNIX_PATH_MAX, "/tmp/demo_socket");

if (bind(socket_fd, (struct sockaddr *) &address, sizeof(struct sockaddr_un)) != 0) {
    printf("bind() failed\n");
    return 1;
}
if(listen(socket_fd, 5) != 0) {
    printf("listen() failed\n");
    return 1;
}

//----------------WHILE LOOP----------------------->
while((connection_fd = accept(socket_fd, 
                            (struct sockaddr *) &address,
                            &address_length)) > -1)
{
.
.
.(doesn't get any further than this)

Это скрипт Python, который я использую для отправки сообщения в сокет:

import socket

s = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
s.connect("/tmp/demo_socket")
print "Sending..."
s.send("Hello world")
x = s.recv(1024)
print x
s.close()

Сценарий python завершается с ошибкой «Broken Pipe». Программа на C никогда не входит в цикл while, потому что функция accept () завершается с ошибкой и возвращает -1.

Почему accept () не работает? И что я могу сделать, чтобы добиться успеха?

Ответы [ 2 ]

2 голосов
/ 25 июля 2011

IIRC, address_length необходимо инициализировать до размера sockaddr_un перед вызовом accept(). Во время вызова он загружается с фактической длиной адреса однорангового узла, который передается в address.

Rgds, Martin

2 голосов
/ 25 июля 2011

Глядя на пример, который я нашел , кажется, что длина, которую вы передаете bind, не должна считать завершающие или заполненные нули в sockaddr_un.

Попробуйте:

size_t addrlen;
const char* sockname = "/tmp/demo_socket";

/* ... */

address.sun_family = AF_UNIX;
snprintf(address.sun_path, UNIX_PATH_MAX, sockname);
addrlen = sizeof(address.sun_family) + strlen(sockname);

if (bind(socket_fd, (struct sockaddr *) &address, addrlen) != 0) {
    printf("bind() failed\n");
    return 1;
}

PS Из-за этого вам не нужен memset вызов.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...