По немногочисленным просьбам трудящихся я расскажу как заворачивать GSM SMS из sip_dongle модуля для Asterisk в sip messaging с последующей маршрутизацией.

Немного ТЗ

Есть центральный узел Asterisk, на котором живет логика и пользователи (extensions). К нему подключаются через PJSIP другие оконечные Asterisk`и на которых висит по несколько GSM модемов.

  • Одна из задач: принимать и отправлять GSM SMS сообщения правильно маршрутизируя их по номеру получателя.

  • Вторая подзадача: дать нашим пользователям (extensions) возможность слать друг другу текстовые сообщения.

Итак. Что тут сложного и в чем подвох?

Когда у вас 1 модем и 1 extension все очень просто. Когда у вас 2 модема и 1 extension задача все еще простая.

Когда у вас “многое ко многим” — вот тут и кроется основная сложность. Т.е. нам нужно не просто понимать на какой конкретно модем какому конкретно Asterisk`у передать исходящее SMS, но и куда направить входящее из GSM сети сообщение. А так же что делать, если мы внутри себя не можем сообщения доставить.

Обозначения:

  • gsmru 10.8.0.11 — оконечный сервер Asterisk на котором живут модемы

  • 10.8.0.10 — центральный сервер Asterisk на котором крутится вся логика

  • 101;102;103 - пользователи (extensions)

  • sip - протокол ходит по 80му порту (так исторически сложилось)

Начнем разворачивать цепочку от оконечного Asterisk (gsmru) с модемами в сторону центрального сервера. Это канал приема сообщений и проще его объяснять именно так.

Работать с SMS можно по разному, но в моем случае я использую sqlite, который у меня /var/lib/asterisk/smsdb.sqlite3. Это сделано так чтобы можно было легко оперировать сообщениями, создавать очереди доставки в обе стороны и не переживать за всякие проблемы вида «я записал что-то в файл и мне нужно успеть это вычитать пока туда не записалось следующее, а заодно проверить что там нет «патча Бармина».

Поэтому в dongle.conf честно вписываем

smsdb=/var/lib/asterisk/smsdb
csmsttl=600


и радуемся результату. Еще не забываем выставить autodeletesms=yes что бы память модема не забивалась и очередь не застревала (мы ж и так все в базу положили).

Как нам организовать БД?

Все просто:

CREATE TABLE incoming (key VARCHAR(256), seqorder INTEGER, expiration TIMESTAMP DEFAULT CURRENT_TIMESTAMP, message VARCHAR(256), PRIMARY KEY(key, seqorder))

CREATE TABLE outgoing_msg (dev VARCHAR(256), dst VARCHAR(255), cnt INTEGER, expiration TIMESTAMP, srr BOOLEAN, payload BLOB)

CREATE TABLE outgoing_part (key VARCHAR(256), msg INTEGER, status INTEGER, PRIMARY KEY(key))

CREATE TABLE outgoing_ref (key VARCHAR(256), refid INTEGER, PRIMARY KEY(key))

CREATE INDEX incoming_key ON incoming(key)

CREATE INDEX outgoing_part_msg ON outgoing_part(msg

На том же самом оконечном Asterisk (gsmru) в extensions.conf создаем правила обработки входящих сообщений.

Напоминаю: 10.8.0.10 — центральный сервер Asterisk, gsmru — наш оконечный узл Asterisk

Прописывам на оконочном Asterisk (gsmru) в extensions.conf:

exten => sms,1,Verbose(Incoming SMS from ${CALLERID(num)} to ${DONGLENAME} ${BASE64_DECODE(${SMS_BASE64})})
same = n,Set(MESSAGE(body)=${BASE64_DECODE(${SMS_BASE64})})
same = n,Set(MSGSENDER=${CALLERID(num)})
same = n,GotoIf($["${DONGLENAME}"="dongle0"]?smsFromDongle0)
same = n,GotoIf($["${DONGLENAME}"="dongle1"]?smsFromDongle1)
same = n,GotoIf($["${DONGLENAME}"="dongle2"]?smsFromDongle2)
same = n(smsFromDongle0),NoOp("Forwarding to 101")
same = n,Set(NDEST=pjsip:gsmru/sip:101@10.8.0.10:80)
same = n,MessageSend(${NDEST},${MSGSENDER})
same = n,Hangup()
same = n(smsFromDongle1),NoOp("Forwarding to 102")
same = n,Set(NDEST=pjsip:gsmru/sip:102@10.8.0.10:80)
same = n,MessageSend(${NDEST},${MSGSENDER})
same = n,Hangup()
same = n(smsFromDongle2),NoOp("Forwarding to 103")
same = n,Set(NDEST=pjsip:gsmru/sip:103@10.8.0.10:80)
same = n,MessageSend(${NDEST},${MSGSENDER})
same = n,Hangup();

Тут все очень просто: сообщаем в консоль что у нас входящее сообщение, печатаем туда же его текст, смотрим от какого устройства и направляем в нужную сторону. Примерно так же можно направить на группу что бы сообщение «разлетелось всем».

Сообщене уже в виде sip message летит на центральный узел 10.8.0.10

Что же там с ним делают?

А там его ждет описание оконечного узла в pjsip.conf

[gsmru]
type=endpoint
transport=transport-tcp
context=incoming-calls
message_context=incoming-sms
disallow=all
allow=ulaw,alaw,opus
outbound_auth=gsmru-auth
aors=gsmru
language=ru

Ну и что бы этот контекст не пропадал даром, в extensions.conf имеем:

;incoming sms from Gsm Gateways
[incoming-sms]
exten =_1.,1,Verbose(1, "Incoming SMS from ${CALLERID(num)} GSM Gateway to ${EXTEN}")
same = n,NoOp(To ${MESSAGE(to)})
same = n,NoOp(From ${MESSAGE(from)})
same = n,NoOp(Body ${MESSAGE(body)})
same = n,MessageSend(${MESSAGE(to)},${MESSAGE(from)})
same = n,NoOp(Send status is ${MESSAGE_SEND_STATUS})
same = n,GotoIf($["${MESSAGE_SEND_STATUS}" != "SUCCESS"]?handlefailedmsg)
same = n,Hangup()

; Handle failed messaging
same = n(handlefailedmsg),NoOp(Sending error to user)
same = n,Set(SRC=${MESSAGE(from)})
same = n,Set(DST=${MESSAGE(to)})
same = n,Set(MSG=${MESSAGE(body)})
same = n,Set(MESSAGE(body)="[${STRFTIME(${EPOCH},,%d%m%Y-%H:%M:%S)}] Your message to ${EXTEN} has failed. Sending when available")
same = n,Set(ME_1=${CUT(MESSAGE(from),<,2)})
same = n,Set(ACTUALFROM=${CUT(ME_1,@,1)})
same = n,MessageSend(${ACTUALFROM},ServiceCenter)
; same = n,GotoIf($["${INQUEUE}" != "1"]?startq)
same = n,Hangup()

Все, магия случилась — сообщение улетело в виде sip mesage в клиентское ПО\Оборудование. Иными словами, если послать sms на номер модема, оно появится в программе. Я уже говорил что не использую аппаратные телефоны, так что тут «не копенгаген», а в Linphone глюк выводит пустое сообщение. Так что на Android я использую PortSip.

Как нам теперь отправлять SMS во внешний мир?

Тут все чуть сложнее и мы пойдем от центрального сервера к модему.

Прописываем на центральном сервере в pjsip.conf:

[endpoint-internal](!)
type = endpoint
context = default
message_context = outgoing-sms
allow = !all,alaw,ulaw,opus,h264,vp8,vp9
direct_media = no
trust_id_outbound = yes
device_state_busy_at = 1
dtmf_mode = rfc4733
media_encryption = sdes
rewrite_contact = yes

Теперь маршрутизация для центрального сервера в extensions.conf:

[outgoing-sms]
exten = _+7.,1,Verbose(1, "Outgoing SMS from ${CALLERID(num)} GSM Gateway to ${EXTEN}")
same = n,NoOp(To ${MESSAGE(to)})
same = n,NoOp(From ${MESSAGE(from)})
same = n,NoOp(Body ${MESSAGE(body)})
same = n,Set(TO= ${MESSAGE(to)})
same = n,Set(BUF= ${CUT(TO,@,1)})
same = n,NoOp(BUF ${BUF})
same = n,Set(NDEST=pjsip:gsmru/sip:${CUT(BUF,:,2)}@10.8.0.11:80)
same = n,NoOp(NDEST ${NDEST})
same = n,Set(BUF1=${CUT(TO,<sip:,2)})
same = n,NoOp(BUF1 ${BUF1})
same = n,Set(BUF2=${STRREPLACE(BUF1:4,>,)})
same = n,NoOp(BUF2 ${BUF2})
same = n,Set(BUF3=${CUT(BUF2,@,1)})
same = n,NoOp(BUF3 ${BUF3})
same = n,GotoIf(${LEN(${BUF3})}=0?:sendsms)
same = n,Set(NDEST=pjsip:gsmru/sip:${BUF3}@10.8.0.11:80)
same = n,NoOp(NDEST ${NDESTD})
same = n(sendsms),MessageSend(${NDEST},${MESSAGE(from)})
same = n,Hangup
exten = _8.,1,Verbose(1, "Outgoing SMS from ${CALLERID(num)} GSM Gateway to ${EXTEN}")
same = n,NoOp(To ${MESSAGE(to)})
same = n,NoOp(From ${MESSAGE(from)})
same = n,NoOp(Body ${MESSAGE(body)})
same = n,Set(TO= ${MESSAGE(to)})
same = n,Set(BUF= ${CUT(TO,@,1)})
same = n,NoOp(BUF ${BUF})
same = n,Set(NDEST=pjsip:gsmru/sip:${CUT(BUF,:,2)}@10.8.0.11:80)
same = n,NoOp(NDEST ${NDEST})
same = n,Set(BUF1=${CUT(TO,<sip:,2)})
same = n,NoOp(BUF1 ${BUF1})
same = n,Set(BUF2=${STRREPLACE(BUF1:4,>,)})
same = n,NoOp(BUF2 ${BUF2})
same = n,Set(BUF3=${CUT(BUF2,@,1)})
same = n,NoOp(BUF3 ${BUF3})
same = n,GotoIf(${LEN(${BUF3})}=0?:sendsms)
same = n,Set(NDEST=pjsip:gsmru/sip:${BUF3}@10.8.0.11:80)
same = n,NoOp(NDEST ${NDESTD})
same = n(sendsms),MessageSend(${NDEST},${MESSAGE(from)})
same = n,Hangup

Сообщение мы смаршрутизировали и даже забуферизировали если не смогли отправить (ну там канал лег или деньги на карточке кончились).

Теперь на нашем оконечном Asterisk gsmru в pjsip.conf создаем описание входящего аккаунта:

[gsmru]
type=endpoint
context=default
message_context=outgoing-sms
disallow=all
allow=ulaw,alaw
transport=transport-tcp
auth=gsmru
aors=gsmru

И на том же оконечном gsmru задаем маршрутизацию в extensions.conf:

[outgoing-sms]
exten = _+X.,1,Verbose(1, "Outgoing SMS from ${CALLERID(num)} to ${EXTEN}.")
same = n,NoOp(To ${MESSAGE(to)})
same = n,NoOp(From ${MESSAGE(from)})
same = n,NoOp(Body ${MESSAGE(body)})
same = n,Set(TO=${MESSAGE(to)})
same = n,Set(FROM=${CUT(MESSAGE(from),<,2)})
same = n,Set(BUF=${CUT(TO,@,1)})
same = n,Set(DEST=${CUT(BUF,:,2)})
same = n,NoOp(DEST=${DEST})
same = n,GotoIf($["${FROM}"="sip:101@SIPDOMAIN>"]?sendSmsDongle0)
same = n,GotoIf($["${FROM}"="sip:102@SIPDOMAIN>"]?sendSmsDongle1)
same = n,GotoIf($["${FROM}"="sip:103@SIPDOMAIN>"]?sendSmsDongle2)
same = n(sendSmsDongle0),DongleSendSMS(dongle0,${DEST},"${MESSAGE(body)}",1140,yes,"magicid")
same = n,Hangup()
same = n(sendSmsDongle1),DongleSendSMS(dongle1,${DEST},"${MESSAGE(body)}",1140,yes,"magicid")
same = n,Hangup()
same = n(sendSmsDongle2),DongleSendSMS(dongle2,${DEST},"${MESSAGE(body)}",1140,yes,"magicid")
same = n,Hangup()
exten = _X.,1,Verbose(1, "Outgoing SMS from ${CALLERID(num)} to ${EXTEN}.")
same = n,NoOp(To ${MESSAGE(to)})
same = n,NoOp(From ${MESSAGE(from)})
same = n,NoOp(Body ${MESSAGE(body)})
same = n,Set(TO=${MESSAGE(to)})
same = n,Set(FROM=${CUT(MESSAGE(from),<,2)})
same = n,Set(BUF=${CUT(TO,@,1)})
same = n,Set(DEST=${CUT(BUF,:,2)})
same = n,NoOp(DEST=${DEST})
same = n,GotoIf($["${FROM}"="sip:101@SIPDOMAIN>"]?sendSmsDongle0)
same = n,GotoIf($["${FROM}"="sip:102@SIPDOMAIN>"]?sendSmsDongle1)
same = n,GotoIf($["${FROM}"="sip:103@SIPDOMAIN>"]?sendSmsDongle2)
same = n(sendSmsDongle0),DongleSendSMS(dongle0,${DEST},"${MESSAGE(body)}",1140,yes,"magicid")
same = n,Hangup()
same = n(sendSmsDongle1),DongleSendSMS(dongle1,${DEST},"${MESSAGE(body)}",1140,yes,"magicid")
same = n,Hangup()
same = n(sendSmsDongle2),DongleSendSMS(dongle2,${DEST},"${MESSAGE(body)}",1140,yes,"magicid")

И вуаля — сообщение уйдет в GSM сеть!

Пару замечаний

  • Я использую PJSIP и ничего не знаю про sip. Ну как бы у нас 2к21 на дворе, пора бы уже двигаться в будущее.

  • Да, я использую транспорт TCP по 80му порту. Это не влияет на рассмотренный вопрос про SMS.

  • Если у вас много стран\городов или много префиксов номера - для них для всех нужно повторять блоки маршрутизации, так же как и для звонков.

  • Обязательна поддержка SRTP на клиенте (оно включено на сервере).

  • На клиенте DTMF нужно выставить в RFC2833.

Комментарии (5)


  1. shachneff
    21.07.2021 19:44

    Посоветуйте, пожалуйста, какой софтфон с поддержкой pjsip на windows можно было бы использовать для описанной схемы? Из реального опыта, в теории и сам знаю


    1. HellKaim Автор
      21.07.2021 20:10

      Даже теоретически не подскажу - нет клиентов на этой ОС. Можно использовать веб версию, если интересно дам ссылку


      1. shachneff
        21.07.2021 21:55

        Спасибо, интересно. Предполагаю, что что-то из этого https://habr.com/ru/post/507554/


        1. HellKaim Автор
          22.07.2021 10:22

          Большинство проектов мертвы. Вот это точно работает

          https://github.com/InnovateAsterisk/Browser-Phone

          У меня был удачный опыт с Bria, но они жадные.


    1. silverjoe
      24.07.2021 22:50

      Да хватит уже путать драйвер PJSIP в Asterisk и протокол SIP!!!

      Протокол не менялся - используйте софтфоны какие хотите - они все работают. Даже 3CXphone 5-6 версии.