Add repro SIP proxy module.

This commit is contained in:
James Valleroy 2016-01-08 19:56:31 -05:00 committed by Sunil Mohan Adapa
parent 863a187b23
commit b7e44eb429
No known key found for this signature in database
GPG Key ID: 36C361440C9BC971
10 changed files with 388 additions and 0 deletions

84
actions/repro Executable file
View File

@ -0,0 +1,84 @@
#!/usr/bin/python3
#
# This file is part of Plinth.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
"""
Configuration helper for repro SIP proxy.
"""
import argparse
from plinth import action_utils
CONFIG = '/etc/repro/repro.config'
def parse_arguments():
"""Return parsed command line arguments as dictionary."""
parser = argparse.ArgumentParser()
subparsers = parser.add_subparsers(dest='subcommand', help='Sub command')
subparsers.add_parser('setup', help='Configure repro')
subparsers.add_parser('enable', help='Enable repro service')
subparsers.add_parser('disable', help='Disable repro service')
return parser.parse_args()
def subcommand_setup(_):
"""Configure repro."""
with open(CONFIG, 'r') as conf:
lines = conf.readlines()
with open(CONFIG, 'w') as conf:
for line in lines:
if line.startswith('Database1Path'):
# workaround for Debian bug #803113
conf.write('Database1Path = /var/lib/repro\n')
elif line.startswith('TLSPort'):
conf.write('TLSPort = 5061\n')
elif line.startswith('DisableHttpAuth'):
# let apache handle authentication
conf.write('DisableHttpAuth = true\n')
else:
conf.write(line)
action_utils.service_restart('repro')
action_utils.webserver_enable('repro-plinth')
def subcommand_enable(_):
"""Start service."""
action_utils.service_enable('repro')
def subcommand_disable(_):
"""Stop service."""
action_utils.service_disable('repro')
def main():
"""Parse arguments and perform all duties."""
arguments = parse_arguments()
subcommand = arguments.subcommand.replace('-', '_')
subcommand_method = globals()['subcommand_' + subcommand]
subcommand_method(arguments)
if __name__ == '__main__':
main()

View File

@ -0,0 +1,11 @@
<Location /repro>
ProxyPass http://localhost:5080
AuthType basic
AuthName "FreedomBox Login"
AuthBasicProvider ldap
AuthLDAPUrl "ldap:///ou=users,dc=thisbox?uid"
AuthLDAPGroupAttribute memberUid
AuthLDAPGroupAttributeIsDN off
Require ldap-group cn=admin,ou=groups,dc=thisbox
</Location>

View File

@ -0,0 +1 @@
plinth.modules.repro

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<service>
<short>repro SIP proxy</short>
<description>repro is a SIP proxy server. It provides authentication, authorization, and call routing features.</description>
<port protocol="udp" port="5060"/>
<port protocol="tcp" port="5060"/>
<port protocol="tcp" port="5061"/>
</service>

View File

@ -0,0 +1,66 @@
#
# This file is part of Plinth.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
"""
Plinth module for repro.
"""
from django.utils.translation import ugettext_lazy as _
from plinth import action_utils
from plinth import cfg
from plinth import service as service_module
depends = ['plinth.modules.apps']
service = None
def init():
"""Initialize the repro module."""
menu = cfg.main_menu.get('apps:index')
menu.add_urlname(_('SIP Proxy (Repro)'), 'glyphicon-phone-alt',
'repro:index', 800)
global service
service = service_module.Service(
'repro', _('Repro SIP Proxy'),
is_external=True, enabled=is_enabled())
def is_enabled():
"""Return whether the service is enabled."""
return action_utils.service_is_enabled('repro')
def is_running():
"""Return whether the service is running."""
return action_utils.service_is_running('repro')
def diagnose():
"""Run diagnostics and return the results."""
results = []
results.append(action_utils.diagnose_port_listening(5060, 'udp4'))
results.append(action_utils.diagnose_port_listening(5060, 'udp6'))
results.append(action_utils.diagnose_port_listening(5060, 'tcp4'))
results.append(action_utils.diagnose_port_listening(5060, 'tcp6'))
results.append(action_utils.diagnose_port_listening(5061, 'tcp4'))
results.append(action_utils.diagnose_port_listening(5061, 'tcp6'))
return results

View File

@ -0,0 +1,30 @@
#
# This file is part of Plinth.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
"""
Forms for repro module.
"""
from django import forms
from django.utils.translation import ugettext_lazy as _
class ReproForm(forms.Form):
"""Configuration form."""
enabled = forms.BooleanField(
label=_('Enable Repro service'),
required=False)

View File

@ -0,0 +1,80 @@
{% extends "base.html" %}
{% comment %}
#
# This file is part of Plinth.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
{% endcomment %}
{% load bootstrap %}
{% load i18n %}
{% block content %}
<h2>{% trans "SIP Proxy (Repro)" %}</h2>
<p>
{% blocktrans trimmed %}
Repro is a SIP proxy service that a SIP softphone can connect to. Before
using Repro, the domain and users will need to be configured using the
web-based configuration panel, available at
<a href="/repro/domains.html">/repro/domains.html</a>. Users in the
<i>admin</i> group will be able to log in to the Repro configuration panel.
{% endblocktrans %}
</p>
<p>
{% blocktrans trimmed %}
<b>Note:</b> After setting the domain, it is required to restart the Repro
service. Disable the service using the checkbox and Update setup button
below. Then, re-enable the service.
{% endblocktrans %}
</p>
<p>
{% blocktrans trimmed %}
To make SIP calls, a client application is needed. Available clients include
<a href="https://jitsi.org/">Jitsi</a> (for computers) and
<a href="https://f-droid.org/repository/browse/?fdid=com.csipsimple">
CSipSimple</a> (for Android phones).
{% endblocktrans %}
</p>
<h3>{% trans "Status" %}</h3>
<p class="running-status-parent">
{% if status.is_running %}
<span class="running-status active"></span>
{% trans "Repro service is running" %}
{% else %}
<span class="running-status inactive"></span>
{% trans "Repro service is not running" %}
{% endif %}
</p>
{% include "diagnostics_button.html" with module="repro" %}
<h3>{% trans "Configuration" %}</h3>
<form class="form" method="post">
{% csrf_token %}
{{ form|bootstrap }}
<input type="submit" class="btn btn-primary"
value="{% trans "Update setup" %}"/>
</form>
{% endblock %}

View File

View File

@ -0,0 +1,29 @@
#
# This file is part of Plinth.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
"""
URLs for the repro module.
"""
from django.conf.urls import url
from . import views
urlpatterns = [
url(r'^apps/repro/$', views.index, name='index'),
]

View File

@ -0,0 +1,79 @@
#
# This file is part of Plinth.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
"""
Views for repro module.
"""
from django.contrib import messages
from django.template.response import TemplateResponse
from django.utils.translation import ugettext as _
from .forms import ReproForm
from plinth import actions
from plinth import package
from plinth.modules import repro
def on_install():
"""Notify that the service is now enabled."""
actions.superuser_run('repro', ['setup'])
repro.service.notify_enabled(None, True)
@package.required(['repro'], on_install=on_install)
def index(request):
"""Serve configuration page."""
status = get_status()
form = None
if request.method == 'POST':
form = ReproForm(request.POST, prefix='repro')
if form.is_valid():
_apply_changes(request, status, form.cleaned_data)
status = get_status()
form = ReproForm(initial=status, prefix='repro')
else:
form = ReproForm(initial=status, prefix='repro')
return TemplateResponse(request, 'repro.html',
{'title': _('SIP Proxy (Repro)'),
'status': status,
'form': form})
def get_status():
"""Get the current service status."""
return {'enabled': repro.is_enabled(),
'is_running': repro.is_running()}
def _apply_changes(request, old_status, new_status):
"""Apply the changes."""
modified = False
if old_status['enabled'] != new_status['enabled']:
sub_command = 'enable' if new_status['enabled'] else 'disable'
actions.superuser_run('repro', [sub_command])
repro.service.notify_enabled(None, new_status['enabled'])
modified = True
if modified:
messages.success(request, _('Configuration updated'))
else:
messages.info(request, _('Setting unchanged'))