Курсы по использованию Asterisk

IP-телефония — технология будущего. Обучитесь работе с IP-АТС Asterisk для того чтобы внедрить и профессионально использовать при решении коммуникационных задач.

Работайте с Asterisk профессионально!

Многоуровневая защита IP-АТС Asterisk

Телефонные станции очень часто становятся объектами хакерских атак. Узнайте, каким образом необходимо строить многоуровневую защиту для Вашей IP-АТС.

Не оставьте хакерам шансов. Защитите свой Asterisk от атак.

Используйте Веб-Интерфейс для удобства настройки

Панель управление FreePBX позволяет легко и удобно управлять всей системой. Научитесь эффективно использовать FreePBX для решения своих задач.

Управление станцией и статистика в окне браузера.

Научитесь работать с Asterisk из консоли

Для понимания работы с Asterisk необходимо уметь настраивать его вручную с конфигурационными файлами и командной строкой CLI Asterisk.

Научитесь «тонкой» настройке Asterisk

Цель курсов - максимум практики.

Обучение нацелено на практическую работу с IP-оборудованием: платы потоков E1, VoIP-телефонные аппараты, голосовые шлюзы FXS и прочее.

Обучение на реальном оборудовании — залог успеха.

Реализация custom callback в elastix.

База знаний Использование Elastix
Данная статья описывает процесс терминации входящего вызова и инициирует обратный звонок пользователю, который звонил только что. Это помогает уменьшит расходы для клиентов, которые, возможно, платят за входящий звонок по тарифам мобильной связи. Исходящий звонок будет совершен по шаблонам и маршрутам, указанным в исходящей маршрутизации.
Другими словами, при входящем звонке в голосовом приветствии абоненту предлагается либо повесить трубку , либо нажать какую-либо клавишу для того чтобы система сама перезвонила абоненту. Для абонента он будет входящим, то есть бесплатным.

Такую схему часто используют различные службы и предприятия (такси, интернет магазины, контакт - центры и проч.) для удержания клиентов, при этом абонент не "висит" долго на линии и не тратит свои деньги на исходящую связь.

В Elastix и Asterisk для этого используются так называемые call-файлы. По сути это обычные текстовые файлы с расширением .call.

Структура этих файлов следующая:
Channel: : Канал, который будет использоваться для исходящего вызова.
CallerID: CallerID содержащий имя и/или номер абонент
MaxRetries: Количество попыток перед тем, как вызов будет считаться неудачным (не включая первую попытку, т.е. 0 = означает совершить 1 попытку вызова). Значение по умолчанию: 0.
RetryTime: Количество секунд между попытками вызова.Значение по умолчанию: 300 (5 минут)
WaitTime: Количество секунд для ожидания ответа на вызов. Значение по умолчанию: 45.
Account: Установка поля “account code”для записи в CDR.
Context: Контекст в файле extensions.conf.
Extension: Название екстеншена в extensions.conf..
Priority: Номер приоритета для екстеншена, с которого нужно начать выполнение.
Set: Установка переменных канала для использования их в логике обработки вызова на заданный екстеншен
Application: Имя приложения Asterisk, которое необходимо выполнить.
Data: Параметры для запускаемого приложения.
AlwaysDelete: Yes/No – Если время модификации .call файла больше текущего, то этот файл не будет удален.
Archive: Yes/No – Переносить или нет .call файл в поддиректорию "outgoing_done" с установленным значением поля "Status: значение", где значение может быть: Completed, Expired или Failed.

Если мы поместим этот файл в каталог /var/spool/asterisk/outgoing, то он будет немедленно исполнен, чего делать нельзя, поэтому создадим отдельный каталог для временного хранения файла и пропишем задания для crontab, чтобы он, например, каждую минуту перемещал call файл на исполнение.

Теперь к практике.
Есть некая очередь 700, есть IVR c голосовым приветствием "...если вы хотите , что бы мы вам перезвонили - нажмите клавишу 0 ..." , необходимо сделать так , чтобы после нажатия 0 астериск сделал 5 попыток дозвона до оператора , при этом уведомив его , что это коллбэк , а не прямой дозвон. После того , как оператор снимет трубку - вызвать внешнего абонента.

Отредактируем файл /eetc/asterisk/extensions_custom.conf ,добавим в него контекст call-back, добавим из файла extensions_additional.conf используюмую IVR, а также контекст в котром располагаются очереди, по умолчанию это контекст ext-queues, также нам нужно создать вспомогательный конекст, который будет использоваться в качестве нашего канала.

Получаем следующее:

[call-back] — это наш основной контекст
exten =>s,1,System(echo -e "Extension: ${CALLERID(num)}\\nCallerID:!!!Call Back!!! <${CALLERID(num)}>\\ nMaxRetries: 10\\nWaitTime: 60000\\nContext: from-internal\\nChannel: Local/s@from-script-n\\nRetryTime: 600" > /var/spool/asterisk/out/${UNIQUEID}.call)

[from-script-n] — вспомогательный контекст для канала
exten = s,1,Goto(ext-queues1,700,1)

[ivr-3] ; Unnamed — наша IVR
include => ivr-3-custom
include => from-did-direct-ivr
exten => 0,1,Goto(call-back,s,1) – здесь указывается какую клавишу нажать и куда перейти
exten => s,n,Set(_IVR_CONTEXT_${CONTEXT}=${IVR_CONTEXT})
exten => s,n,Set(_IVR_CONTEXT=${CONTEXT})
exten => s,n,Set(__IVR_RETVM=)
exten => s,n,GotoIf($["${CDR(disposition)}" = "ANSWERED"]?skip)
exten => s,n,Answer
exten => s,n,Wait(1)
exten => s,n(skip),Set(IVR_MSG=)
exten => s,n(start),Set(TIMEOUT(digit)=3)
exten => s,n,ExecIf($["${IVR_MSG}" != ""]?Background(${IVR_MSG}))
exten => s,n,WaitExten(10,)
exten => i,1,Playback(sorry-youre-having-problems)
exten => i,n,Goto(hang,1)
exten => t,1,Playback(sorry-youre-having-problems)
exten => t,n,Goto(hang,1)
exten => return,1,Set(_IVR_CONTEXT=${CONTEXT})
exten => return,n,Set(_IVR_CONTEXT_${CONTEXT}=${IVR_CONTEXT_${CONTEXT}})
exten => return,n,Set(IVR_MSG=)
exten => return,n,Goto(s,start)
exten => h,1,Hangup
exten => hang,1,Playback(vm-goodbye)
exten => hang,n,Hangup
;--== end of [ivr-3] ==--;

[ext-queues1]
include => ext-queues-custom
exten => 700,1,Macro(user-callerid,)
exten => 700,n,Answer
exten => 700,n,Macro(blkvm-set,reset)
exten => 700,n,ExecIf($["${REGEX("(M[(]auto-blkvm[)])" ${DIAL_OPTIONS})}" != "1"]?Set(_DIAL_OPTIONS=${DIAL_OPTIONS}M(auto-blkvm)))
exten => 700,n,Set(__NODEST=${EXTEN})
exten => 700,n,Set(QCIDPP=${IF($[${LEN(${VQ_CIDPP})}>0]?${VQ_CIDPP}: )})
exten => 700,n,Set(VQ_CIDPP=)
exten => 700,n,ExecIf($["${QCIDPP}"!=""]?Macro(prepend-cid,${QCIDPP}))
exten => 700,n,Set(QAINFO=${IF($[${LEN(${VQ_AINFO})}>0]?${VQ_AINFO}: )})
exten => 700,n,Set(VQ_AINFO=)
exten => 700,n,ExecIf($["${QAINFO}"!=""]?Set(__ALERT_INFO=${QAINFO}))
exten => 700,n,Set(QJOINMSG=${IF($[${LEN(${VQ_JOINMSG})}>0]?${IF($["${VQ_JOINMSG}"!="0"]?${VQ_JOINMSG}: )}: )})
exten => 700,n,Set(VQ_JOINMSG=)
exten => 700,n,Set(QMOH=${IF($["${VQ_MOH}"!=""]?${VQ_MOH}:none)})
exten => 700,n,Set(VQ_MOH=)
exten => 700,n,ExecIf($["${QMOH}"!=""]?Set(__MOHCLASS=${QMOH}))
exten => 700,n,ExecIf($["${MOHCLASS}"!=""]?Set(CHANNEL(musicclass)=${MOHCLASS}))
exten => 700,n,Set(QCANCELMISSED=C)
exten => 700,n,Set(QRINGOPTS=r)
exten => 700,n,Set(QRETRY=${IF($[${LEN(${VQ_RETRY})}>0]?${VQ_RETRY}: )})
exten => 700,n,Set(VQ_RETRY=)
exten => 700,n(qoptions),Set(QOPTIONS=${IF($[${LEN(${VQ_OPTIONS})}>0]?${VQ_OPTIONS}:t)}${QCANCELMISSED}${QRINGOPTS}${QRETRY})
exten => 700,n,Set(VQ_OPTIONS=)
exten => 700,n(qgosub),Set(QGOSUB=${IF($[${LEN(${VQ_GOSUB})}>0]?${VQ_GOSUB}:${QGOSUB})})
exten => 700,n,Set(VQ_GOSUB=)
exten => 700,n(qagi),Set(QAGI=${IF($[${LEN(${VQ_AGI})}>0]?${VQ_AGI}:${QAGI})})
exten => 700,n,Set(VQ_AGI=)
exten => 700,n(qrule),Set(QRULE=${IF($[${LEN(${VQ_RULE})}>0]?${IF($["${VQ_RULE}"!="0"]?${VQ_RULE}: )}:${QRULE})})
exten => 700,n,Set(VQ_RULE=)
exten => 700,n(qposition),Set(QPOSITION=${IF($[${LEN(${VQ_POSITION})}>0]?${VQ_POSITION}:${QPOSITION})})
exten => 700,n,Set(VQ_POSITION=)
exten => 700,n,Set(MONITOR_OPTIONS=b)
exten => 700,n,Gosub(sub-record-check,s,1(q,700,dontcare))
exten => 700,n,Set(__CWIGNORE=TRUE)
exten => 700,n,ExecIf($["${QJOINMSG}"!=""]?Playback(${QJOINMSG}, ))
exten => 700,n,QueueLog(700,${UNIQUEID},NONE,DID,${FROM_DID})
exten => 700,n,Set(QAANNOUNCE=${IF($[${LEN(${VQ_AANNOUNCE})}>0]?${IF($["${VQ_AANNOUNCE}"!="0"]?${VQ_AANNOUNCE}: )}:custom/priv-trying )})
exten => 700,n,Set(VQ_AANNOUNCE=)
exten => 700,n,Set(QMAXWAIT=${IF($[${LEN(${VQ_MAXWAIT})}>0]?${VQ_MAXWAIT}: )})
exten => 700,n,Set(VQ_MAXWAIT=)
exten => 700,n,Set(QUEUENUM=700)
exten => 700,n,Set(QUEUEJOINTIME=${EPOCH})
exten => 700,n(qcall),Queue(700,${QOPTIONS},,${QAANNOUNCE},${QMAXWAIT},${QAGI},,${QGOSUB},${QRULE},${QPOSITION})
exten => 700,n,Macro(blkvm-clr,)
exten => 700,n,Gosub(sub-record-cancel,s,1())
exten => 700,n,Set(__NODEST=)
exten => 700,n,Set(_QUEUE_PRIO=0)
exten => 700,n,Set(MONITOR_OPTIONS=)
exten => 700,n,Set(__CWIGNORE=)
exten => 700,n,Set(QRINGOPTS=)
exten => 700,n,Set(QDEST=${VQ_DEST})
exten => 700,n,Set(VQ_DEST=)
exten => 700,n(gotodest),GotoIf($["${QDEST}"=""]?ext-queues,400,1:${CUT(QDEST,^,1)},${CUT(QDEST,^,2)},${CUT(QDEST,^,3)})
exten => 700*,1,Macro(agent-add,700,)
exten => 700**,1,Macro(agent-del,700)

SettingsВ самом elastix в настройках очереди необходимо указать следующий параметр
Для того чтобы клиентам приходило сообщение о возможности заказа обратного звонка.

И последнее что нам осталось сделать — это дать задание для crontab

crontab -e
*/1 * * * * mv /var/spool/asterisk/out/*.call /var/spool/asterisk/outgoing/

asterisk, установка, Call-файл, Channel, call, callerid, IVR, оператор связи, Elastix, 1C