Как отправлять многоадресные сообщения и повторно использовать порт в Erlang? - PullRequest
7 голосов
/ 17 ноября 2009

Я хорошо начал свою программу, мою первую программу REAL Erlang. Я слушаю сообщения, читаю и разбираю их. У меня также есть его отправка. Одна маленькая вещь, которая беспокоит меня, это то, что я не могу отправить на порт 5353, я все перепробовал. Все остальные приложения на моем компьютере могут прослушивать и отправлять через порт 5353, SubEthaEdit, iTunes, iChat.

Решение ДОЛЖНО транслировать передачу на порт 5353, и вот почему.

"Если исходный UDP-порт в полученном многоадресном DNS-запросе не является портом 5353, это указывает, что клиент, инициирующий запрос, является простой клиент, который не полностью реализует все Multicast DNS. В этом случае многоадресный DNS-ответчик ДОЛЖЕН отправить UDP-ответ непосредственно обратно к клиенту, через одноадресную передачу, к пакету запроса IP-адрес источника и порт. Этот одноадресный ответ ДОЛЖЕН быть обычный одноадресный ответ, который будет генерироваться обычным одноадресный DNS-сервер; например, он ДОЛЖЕН повторить идентификатор запроса и вопрос задан в пакете запроса. «

Все они сообщают порт: 5353 при отправке многоадресных сообщений. Я действительно хочу, чтобы мое приложение играло хорошо и делало то же самое, отправьте на порт 5353. Вот мой модуль в его нынешнем виде.

-module(zeroconf).

-include("zeroconf.hrl").

-export([open/0,start/0]).
-export([stop/1,receiver/0]).
-export([send/1]).

-define(ADDR, {224,0,0,251}).
-define(PORT, 5353).

send(Domain) ->
    {ok,S} = gen_udp:open(0,[{broadcast,true}]), % I really want this Port to be 5353 :-(
    % this doesn't complain or throw errors but it also doesn't work :-(        
    %{ok,S} = gen_udp:open(?PORT,[{reuseaddr,true}, {ip,?ADDR}, {broadcast,true},multicast_ttl,4}, {multicast_loop,false}, binary]),
    P = #dns_rec{header=#dns_header{},qdlist=[#dns_query{domain=Domain,type=ptr,class=in}]},
    gen_udp:send(S,?ADDR,?PORT,inet_dns:encode(P)),
    gen_udp:close(S).

Вот как выглядит какой-то вывод.

Это ЗАПРОС от SubEthaEdit, который ищет другие экземпляры в локальной сети, обратите внимание, что он говорит: Порт: 5353

From: {192,168,0,105}
Port: 5353
Data: {ok,{dns_rec,{dns_header,0,true,'query',true,false,false,false,false,0},
                   [],
                   [{dns_rr,"_see._tcp.local",ptr,in,0,0,
                            "jhr@Blackintosh._see._tcp.local",undefined,[],
                            false}],
                   [],[]}}

Теперь вот запрос из моего модуля, который ищет экземпляры iTunes в локальной сети. Обратите внимание: Port: 59795 С кодом, как сейчас, этот порт является случайным. Я действительно хочу, чтобы это было 5353.

From: {192,168,0,105}
Port: 59795
Data: {ok,{dns_rec,{dns_header,0,false,'query',false,false,false,false,false,
                               0},
                   [{dns_query,"_daap._tcp.local",ptr,in}],
                   [],[],[]}}

Есть ли у кого-нибудь какое-то загадочное понимание многоадресной рассылки UDP? Обновление, чтобы я мог попытаться принять ответ. Я думаю, что просто не могу этого сделать.

Ответы [ 3 ]

3 голосов
/ 17 ноября 2009

ОБНОВЛЕНО : хорошо, я нашел то, что я считаю рабочим решением. Кажется, что критический момент связан с присоединением к многоадресной группе .

{ok, Socket} = gen_udp:open(Port=5353, [binary, {active, false}, {reuseaddr, true},
                                        {ip, Addr}, {add_membership, {Addr, IAddr}}]).
  1. Адрес: многоадресная группа (например, {224, 0, 0, 251}
  2. IAddr - это локальный IP-интерфейс (например, можно использовать значение по умолчанию {0,0,0,0})

(Конечно, убедитесь, что у вас не запущен демон DNS, который может вступить в конфликт)

1 голос
/ 05 декабря 2009

не хватает представителя для ответа на {трансляцию, правда} в сообщении Эмиля, извините.

Должен быть установлен флаг сокета SO_BROADCAST (который, как я предполагаю, сопоставляется), иначе sendto (широковещательный адрес) не будет выполнен. Это предохранитель для предотвращения злоупотреблений или ошибок в программах, которые не предназначены для трансляции. В противном случае защищенным программам придется самим проверять широковещательные адреса.

включение SO_BROADCAST не останавливает вас от отправки не вещательных пакетов. (опять же, если предположить, что материал erlang просто сопоставляется напрямую с setsockopts; я не знаю erlang, просто работаю в сети!)

Возможно, вы захотите попробовать strace, чтобы увидеть, какие системные вызовы действительно происходят. ищите socket (), а затем, что происходит с этим дескриптором файла.

0 голосов
/ 17 ноября 2009

Вы пытаетесь открыть сокет, который уже открыт? Разве вы не можете использовать один и тот же сокет для отправки и получения?

...