views.py 32.8 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
# Copyright 2014 Budapest University of Technology and Economics (BME IK)
#
# This file is part of CIRCLE Cloud.
#
# CIRCLE is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free
# Software Foundation, either version 3 of the License, or (at your option)
# any later version.
#
# CIRCLE 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 General Public License for more
# details.
#
# You should have received a copy of the GNU General Public License along
# with CIRCLE.  If not, see <http://www.gnu.org/licenses/>.

18
from netaddr import IPNetwork
19 20
from django.views.generic import (TemplateView, UpdateView, DeleteView,
                                  CreateView)
21
from django.core.exceptions import ValidationError
22
from django.core.urlresolvers import reverse_lazy
23
from django.shortcuts import render, redirect, get_object_or_404
24
from django.http import HttpResponse, Http404
25 26 27

from django_tables2 import SingleTableView

28
from firewall.models import (Host, Vlan, Domain, Group, Record, BlacklistItem,
29
                             Rule, VlanGroup, SwitchPort, EthernetDevice)
30
from vm.models import Interface
31
from .tables import (HostTable, VlanTable, SmallHostTable, DomainTable,
32
                     GroupTable, RecordTable, BlacklistItemTable, RuleTable,
33
                     VlanGroupTable, SmallRuleTable, SmallGroupRuleTable,
34
                     SmallRecordTable, SwitchPortTable, SmallDhcpTable, )
35
from .forms import (HostForm, VlanForm, DomainForm, GroupForm, RecordForm,
36
                    BlacklistItemForm, RuleForm, VlanGroupForm, SwitchPortForm)
37 38

from django.contrib import messages
39
from django.contrib.messages.views import SuccessMessageMixin
40 41
from django.views.generic.edit import FormMixin
from django.utils.translation import ugettext_lazy as _
42
from braces.views import LoginRequiredMixin, SuperuserRequiredMixin
43 44
# from django.db.models import Q
from operator import itemgetter
45 46
from itertools import chain
import json
Kálmán Viktor committed
47
from dashboard.views import AclUpdateView
48
from dashboard.forms import AclUserOrGroupAddForm
49

50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
from django.utils import simplejson

try:
    from django.http import JsonResponse
except ImportError:
    class JsonResponse(HttpResponse):
        """JSON response for Django < 1.7
        https://gist.github.com/philippeowagner/3179eb475fe1795d6515
        """
        def __init__(self, content, mimetype='application/json',
                     status=None, content_type=None):
            super(JsonResponse, self).__init__(
                content=simplejson.dumps(content),
                mimetype=mimetype,
                status=status,
                content_type=content_type)

67

68 69 70 71 72
class InitialOwnerMixin(FormMixin):
    def get_initial(self):
        initial = super(InitialOwnerMixin, self).get_initial()
        initial['owner'] = self.request.user
        return initial
73 74


75
class IndexView(LoginRequiredMixin, SuperuserRequiredMixin, TemplateView):
76 77 78 79 80 81
    template_name = "network/index.html"

    def get_context_data(self, **kwargs):
        context = super(IndexView, self).get_context_data(**kwargs)

        size = 13
82 83
        blacklists = BlacklistItem.objects.all().order_by(
            '-modified_at')[:size]
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
        domains = Domain.objects.all().order_by('-modified_at')[:size]
        groups = Group.objects.all().order_by('-modified_at')[:size]
        hosts = Host.objects.all().order_by('-modified_at')[:size]
        records = Record.objects.all().order_by('-modified_at')[:size]
        vlans = Vlan.objects.all().order_by('-modified_at')[:size]
        vlangroups = VlanGroup.objects.all().order_by('-modified_at')[:size]
        rules = Rule.objects.all().order_by('-modified_at')[:size]

        result_list = []
        for i in (sorted(chain(domains, groups, hosts, records, vlans,
                               vlangroups, rules),
                         key=lambda x: x.modified_at, reverse=True)[:size]):
            result_list.append(
                {
                    'class_name': unicode(i.__class__.__name__),
                    'modified_at': i.modified_at,
                    'created_at': i.created_at,
                    'name': unicode(i),
                    'link': i.get_absolute_url()
                })

        context['latest_blacklists'] = blacklists
        context['latest'] = result_list
        return context


110 111
class BlacklistList(LoginRequiredMixin, SuperuserRequiredMixin,
                    SingleTableView):
112 113
    model = BlacklistItem
    table_class = BlacklistItemTable
114 115 116 117
    template_name = "network/blacklist-list.html"
    table_pagination = False


118 119
class BlacklistDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                      SuccessMessageMixin, UpdateView):
120
    model = BlacklistItem
121
    template_name = "network/blacklist-edit.html"
122 123
    form_class = BlacklistItemForm
    success_message = _(u'Successfully modified blacklist item'
124 125 126 127 128 129 130 131 132 133 134 135
                        '%(ipv4)s - %(type)s!')

    def get_success_url(self):
        if 'pk' in self.kwargs:
            return reverse_lazy('network.blacklist', kwargs=self.kwargs)

    def get_context_data(self, **kwargs):
        context = super(BlacklistDetail, self).get_context_data(**kwargs)
        context['blacklist_pk'] = self.object.pk
        return context


136 137
class BlacklistCreate(LoginRequiredMixin, SuperuserRequiredMixin,
                      SuccessMessageMixin, CreateView):
138
    model = BlacklistItem
139
    template_name = "network/blacklist-create.html"
140 141
    form_class = BlacklistItemForm
    success_message = _(u'Successfully created blacklist item '
142 143 144
                        '%(ipv4)s - %(type)s!')


145
class BlacklistDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
146
    model = BlacklistItem
147 148 149 150 151 152
    template_name = "network/confirm/base_delete.html"

    def get_context_data(self, **kwargs):
        """ display more information about the object """
        context = super(BlacklistDelete, self).get_context_data(**kwargs)
        if 'pk' in self.kwargs:
153
            to_delete = BlacklistItem.objects.get(pk=self.kwargs['pk'])
154 155 156 157 158 159 160 161 162 163 164 165 166
            context['object'] = "%s - %s - %s" % (to_delete.ipv4,
                                                  to_delete.reason,
                                                  to_delete.type)
            return context

    def get_success_url(self):
        next = self.request.POST.get('next')
        if next:
            return self.request.POST['next']
        else:
            return reverse_lazy('network.blacklist_list')


167
class DomainList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
168 169 170 171 172 173
    model = Domain
    table_class = DomainTable
    template_name = "network/domain-list.html"
    table_pagination = False


174 175
class DomainDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                   SuccessMessageMixin, UpdateView):
176 177 178
    model = Domain
    template_name = "network/domain-edit.html"
    form_class = DomainForm
179
    success_message = _(u'Successfully modified domain %(name)s!')
180 181 182 183 184 185 186

    def get_success_url(self):
        if 'pk' in self.kwargs:
            return reverse_lazy('network.domain', kwargs=self.kwargs)

    def get_context_data(self, *args, **kwargs):
        context = super(DomainDetail, self).get_context_data(**kwargs)
187 188 189 190 191 192 193
        self.object = self.get_object()
        context['domain_pk'] = self.object.pk

        q = Record.objects.filter(
            domain=self.object,
            host__in=Host.objects.filter(
                interface__in=Interface.objects.filter(
194
                    instance__destroyed_at=None)
195 196
            )
        )
197 198 199 200
        context['record_list'] = SmallRecordTable(q)
        return context


201
class DomainCreate(LoginRequiredMixin, SuperuserRequiredMixin,
202
                   SuccessMessageMixin, InitialOwnerMixin, CreateView):
203 204 205
    model = Domain
    template_name = "network/domain-create.html"
    form_class = DomainForm
206
    success_message = _(u'Successfully created domain %(name)s!')
207 208


209
class DomainDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
210 211 212 213 214 215 216 217 218 219 220 221 222
    model = Domain
    template_name = "network/confirm/base_delete.html"

    def get_success_url(self):
        next = self.request.POST.get('next')
        if next:
            return self.request.POST['next']
        else:
            return reverse_lazy('network.domain_list')

    def delete(self, request, *args, **kwargs):
        self.object = self.get_object()
        if unicode(self.object) != request.POST.get('confirm'):
223
            messages.error(request, _(u"Object name does not match!"))
224 225 226
            return self.get(request, *args, **kwargs)

        response = super(DomainDelete, self).delete(request, *args, **kwargs)
227
        messages.success(request, _(u"Domain successfully deleted!"))
228 229 230 231 232
        return response

    def get_context_data(self, **kwargs):
        context = super(DomainDelete, self).get_context_data(**kwargs)

233
        records_from_hosts = _(u'Records from hosts')
234 235 236 237 238
        deps = []
        # vlans
        vlans = Vlan.objects.filter(domain=self.object).all()
        if len(vlans) > 0:
            deps.append({
239
                'name': _('Vlans'),
240 241 242 243 244 245 246
                'data': vlans
            })

            # hosts
            hosts = Host.objects.filter(vlan__in=deps[0]['data'])
            if len(hosts) > 0:
                deps.append({
247
                    'name': _('Hosts'),
248 249 250 251 252
                    'data':  hosts
                })

                # records
                records = Record.objects.filter(
253 254
                    host__in=deps[1]['data']
                    # Q(domain=self.object) | (host__in=deps[1]['data'])
255 256 257
                )
                if len(records) > 0:
                    deps.append({
258
                        'name': records_from_hosts,
259 260 261
                        'data': records
                    })

262 263 264 265
        records = Record.objects.filter(domain=self.object)
        if len(records) > 0:
            # to filter out doubles (records from hosts and domains)
            indexes = map(itemgetter('name'), deps)
266
            n = indexes.index(records_from_hosts) if len(indexes) > 0 else 0
267
            deps.append({
268
                'name': u'Records only from the domain',
269 270 271 272
                'data': records.exclude(pk__in=deps[n]['data']) if n > 0
                else records
            })

273 274 275 276 277
        context['deps'] = deps
        context['confirmation'] = True
        return context


278
class GroupList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
279 280 281 282 283 284
    model = Group
    table_class = GroupTable
    template_name = "network/group-list.html"
    table_pagination = False


285
class GroupCreate(LoginRequiredMixin, SuperuserRequiredMixin,
286
                  SuccessMessageMixin, InitialOwnerMixin, CreateView):
287 288 289
    model = Group
    template_name = "network/group-create.html"
    form_class = GroupForm
290
    success_message = _(u'Successfully created host group %(name)s!')
291 292


293 294
class GroupDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                  SuccessMessageMixin, UpdateView):
295 296 297
    model = Group
    template_name = "network/group-edit.html"
    form_class = GroupForm
298
    success_message = _(u'Successfully modified host group %(name)s!')
299 300 301 302 303

    def get_success_url(self):
        if 'pk' in self.kwargs:
            return reverse_lazy('network.group', kwargs=self.kwargs)

304 305 306
    def get_context_data(self, *args, **kwargs):
        context = super(GroupDetail, self).get_context_data(**kwargs)

307 308
        context['group_pk'] = self.object.pk

309 310 311 312 313
        # records
        q = Rule.objects.filter(hostgroup=self.object)
        context['rule_list'] = SmallRuleTable(q)
        return context

314

315
class GroupDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331
    model = Group
    template_name = "network/confirm/base_delete.html"

    def get_success_url(self):
        next = self.request.POST.get('next')
        if next:
            return self.request.POST['next']
        else:
            return reverse_lazy('network.group_list')

    def get_context_data(self, *args, **kwargs):
        context = super(GroupDelete, self).get_context_data(**kwargs)
        context['group_pk'] = self.object.pk
        return context


332
class HostList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
333 334 335 336 337 338 339 340 341 342 343 344 345 346
    model = Host
    table_class = HostTable
    template_name = "network/host-list.html"
    table_pagination = False

    def get_context_data(self, **kwargs):
        context = super(HostList, self).get_context_data(**kwargs)
        q = Vlan.objects.all().order_by("name")
        context['vlans'] = q
        return context

    def get_table_data(self):
        vlan_id = self.request.GET.get('vlan')
        if vlan_id:
347
            data = Host.objects.filter(vlan=vlan_id).select_related()
348
        else:
349
            data = Host.objects.select_related()
350 351 352
        return data


353 354
class HostDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                 SuccessMessageMixin, UpdateView):
355 356 357
    model = Host
    template_name = "network/host-edit.html"
    form_class = HostForm
358
    success_message = _(u'Successfully modified host %(hostname)s!')
359 360 361 362 363 364

    def get(self, request, *args, **kwargs):
        if request.is_ajax():
            host = Host.objects.get(pk=kwargs['pk'])
            host = {
                'hostname': host.hostname,
365 366
                'ipv4': str(host.ipv4),
                'ipv6': str(host.ipv6),
367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410
                'fqdn': host.get_fqdn()
            }
            return HttpResponse(json.dumps(host),
                                content_type="application/json")
        else:
            self.object = self.get_object()
            return super(HostDetail, self).get(request, *args, **kwargs)

    def post(self, request, *args, **kwargs):
        pk = self.kwargs.get('pk')
        # this is usually not None (well, with curl and whatnot it can be)
        if pk:
            groups = Host.objects.get(pk=pk).groups.all()
            groups = [i.pk for i in groups]
            # request.POST is immutable
            post_copy = request.POST.copy()
            post_copy.setlist('groups', groups)
            request.POST = post_copy
            return super(HostDetail, self).post(request, *args, **kwargs)

    def get_context_data(self, **kwargs):
        context = super(HostDetail, self).get_context_data(**kwargs)
        # own rules
        q = Rule.objects.filter(host=self.object).all()
        context['rule_list'] = SmallRuleTable(q)

        # rules from host groups
        group_rule_list = []
        for group in self.object.groups.all():
            q = Rule.objects.filter(hostgroup=group).all()
            group_rule_list.append({
                'table': SmallGroupRuleTable(q),
                'name': unicode(group),
                'pk': group.pk
            })
        context['group_rule_list'] = group_rule_list

        # available groups
        rest = Group.objects.exclude(pk__in=self.object.groups.all()).all()
        context['not_used_groups'] = rest

        # set host pk (we need this for URL-s)
        context['host_pk'] = self.kwargs['pk']

411 412 413
        from network.tables import HostRecordsTable
        context['records_table'] = HostRecordsTable(
            Record.objects.filter(host=self.get_object()),
414
            request=self.request, template="django_tables2/table_no_page.html"
415 416
        )

417 418 419 420 421 422 423
        return context

    def get_success_url(self):
        if 'pk' in self.kwargs:
            return reverse_lazy('network.host', kwargs=self.kwargs)


424
class HostCreate(LoginRequiredMixin, SuperuserRequiredMixin,
425
                 SuccessMessageMixin, InitialOwnerMixin, CreateView):
426 427 428
    model = Host
    template_name = "network/host-create.html"
    form_class = HostForm
429
    success_message = _(u'Successfully created host %(hostname)s!')
430

431 432 433 434 435 436 437
    def get_initial(self):
        initial = super(HostCreate, self).get_initial()

        for i in ("vlan", "mac", "hostname"):
            if i in self.request.GET and i not in self.request.POST:
                initial[i] = self.request.GET[i]
        if "vlan" in initial:
438 439
            if not initial['vlan'].isnumeric():
                raise Http404()
440 441 442 443 444 445 446
            vlan = get_object_or_404(Vlan.objects, pk=initial['vlan'])
            try:
                initial.update(vlan.get_new_address())
            except ValidationError as e:
                messages.error(self.request, e.message)
        return initial

447 448 449
    def _get_ajax(self, *args, **kwargs):
        GET = self.request.GET
        result = {}
450 451 452 453 454 455 456
        vlan = get_object_or_404(Vlan.objects, pk=GET.get("vlan", ""))
        if "ipv4" in GET:
            try:
                result["ipv6"] = vlan.convert_ipv4_to_ipv6(GET["ipv4"])
            except:
                result["ipv6"] = ""
        else:
457 458 459 460 461
            try:
                result.update(vlan.get_new_address())
            except ValidationError:
                result["ipv4"] = ""
                result["ipv6"] = ""
462
        return JsonResponse({k: unicode(result[k] or "") for k in result})
463 464 465 466 467 468 469

    def get(self, *args, **kwargs):
        if self.request.is_ajax():
            return self._get_ajax(*args, **kwargs)
        else:
            return super(HostCreate, self).get(*args, **kwargs)

470

471
class HostDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488
    model = Host
    template_name = "network/confirm/base_delete.html"

    def get_success_url(self):
        next = self.request.POST.get('next')
        if next:
            return self.request.POST['next']
        else:
            return reverse_lazy('network.host_list')

    def get_context_data(self, *args, **kwargs):
        context = super(HostDelete, self).get_context_data(**kwargs)

        deps = []
        records = Record.objects.filter(host=self.object).all()
        if records:
            deps.append({
489
                'name': _('Records'),
490 491 492 493 494 495 496 497 498 499
                'data': records
            })

        context['deps'] = deps
        context['confirmation'] = True
        return context

    def delete(self, request, *args, **kwargs):
        self.object = self.get_object()
        if unicode(self.object) != request.POST.get('confirm'):
500
            messages.error(request, _(u"Object name does not match!"))
501 502 503
            return self.get(request, *args, **kwargs)

        response = super(HostDelete, self).delete(request, *args, **kwargs)
504
        messages.success(request, _(u"Host successfully deleted!"))
505 506 507
        return response


508
class RecordList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
509 510 511 512 513
    model = Record
    table_class = RecordTable
    template_name = "network/record-list.html"
    table_pagination = False

514 515 516 517 518 519 520 521 522 523 524 525 526
    def get_context_data(self, **kwargs):
        context = super(RecordList, self).get_context_data(**kwargs)
        context['types'] = Record.CHOICES_type
        return context

    def get_table_data(self):
        type_id = self.request.GET.get('type')
        if type_id:
            data = Record.objects.filter(type=type_id).select_related()
        else:
            data = Record.objects.select_related()
        return data

527

528 529
class RecordDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                   SuccessMessageMixin, UpdateView):
530 531 532 533
    model = Record
    template_name = "network/record-edit.html"
    form_class = RecordForm
    # TODO fqdn
534
    success_message = _(u'Successfully modified record!')
535 536 537 538 539 540 541 542 543 544 545 546

    def get_context_data(self, **kwargs):
        context = super(RecordDetail, self).get_context_data(**kwargs)
        context['fqdn'] = self.object.fqdn
        context['record_pk'] = self.object.pk
        return context

    def get_success_url(self):
        if 'pk' in self.kwargs:
            return reverse_lazy('network.record', kwargs=self.kwargs)


547
class RecordCreate(LoginRequiredMixin, SuperuserRequiredMixin,
548
                   SuccessMessageMixin, InitialOwnerMixin, CreateView):
549 550 551 552
    model = Record
    template_name = "network/record-create.html"
    form_class = RecordForm
    # TODO fqdn
553
    success_message = _(u'Successfully created record!')
554

555
    def get_initial(self):
556 557 558
        initial = super(RecordCreate, self).get_initial()
        initial['domain'] = self.request.GET.get('domain')

559 560 561 562 563 564 565 566 567 568 569 570 571
        host_pk = self.request.GET.get("host")
        try:
            host = Host.objects.get(pk=host_pk)
        except (Host.DoesNotExist, ValueError):
            host = None

        if host:
            initial.update({
                'type': "CNAME",
                'host': host,
                'address': host.get_fqdn(),
            })

572
        return initial
573

574

575
class RecordDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
576 577 578 579 580 581 582 583 584 585 586
    model = Record
    template_name = "network/confirm/base_delete.html"

    def get_success_url(self):
        next = self.request.POST.get('next')
        if next:
            return self.request.POST['next']
        else:
            return reverse_lazy('network.record_list')


587
class RuleList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
588 589 590 591 592
    model = Rule
    table_class = RuleTable
    template_name = "network/rule-list.html"
    table_pagination = False

593 594 595 596 597
    def get_table_data(self):
        return Rule.objects.select_related('host', 'hostgroup', 'vlan',
                                           'vlangroup', 'firewall',
                                           'foreign_network', 'owner')

598

599 600
class RuleDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                 SuccessMessageMixin, UpdateView):
601 602 603
    model = Rule
    template_name = "network/rule-edit.html"
    form_class = RuleForm
604
    success_message = _(u'Successfully modified rule!')
605 606 607 608 609 610 611 612 613 614 615 616 617 618

    def get_success_url(self):
        if 'pk' in self.kwargs:
            return reverse_lazy('network.rule', kwargs=self.kwargs)

    def get_context_data(self, **kwargs):
        context = super(RuleDetail, self).get_context_data(**kwargs)

        rule = self.get_object()

        context['rule'] = rule
        return context


619
class RuleCreate(LoginRequiredMixin, SuperuserRequiredMixin,
620
                 SuccessMessageMixin, InitialOwnerMixin, CreateView):
621 622 623
    model = Rule
    template_name = "network/rule-create.html"
    form_class = RuleForm
624
    success_message = _(u'Successfully created rule!')
625

626
    def get_initial(self):
627 628
        initial = super(RuleCreate, self).get_initial()
        initial.update({
629 630
            'host': self.request.GET.get('host'),
            'hostgroup': self.request.GET.get('hostgroup')
631 632
        })
        return initial
633

634

635
class RuleDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
636 637 638 639 640 641 642 643 644 645 646
    model = Rule
    template_name = "network/confirm/base_delete.html"

    def get_success_url(self):
        next = self.request.POST.get('next')
        if next:
            return next
        else:
            return reverse_lazy('network.rule_list')


647 648
class SwitchPortList(LoginRequiredMixin, SuperuserRequiredMixin,
                     SingleTableView):
649 650 651 652 653 654
    model = SwitchPort
    table_class = SwitchPortTable
    template_name = "network/switch-port-list.html"
    table_pagination = False


655 656
class SwitchPortDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                       SuccessMessageMixin, UpdateView):
657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673
    model = SwitchPort
    template_name = "network/switch-port-edit.html"
    form_class = SwitchPortForm
    success_message = _(u'Succesfully modified switch port!')

    def get_success_url(self):
        if 'pk' in self.kwargs:
            return reverse_lazy('network.switch_port', kwargs=self.kwargs)

    def get_context_data(self, **kwargs):
        context = super(SwitchPortDetail, self).get_context_data(**kwargs)
        context['switch_port_pk'] = self.object.pk
        context['devices'] = EthernetDevice.objects.filter(
            switch_port=self.object.pk)
        return context


674 675
class SwitchPortCreate(LoginRequiredMixin, SuperuserRequiredMixin,
                       SuccessMessageMixin, CreateView):
676 677 678 679 680 681
    model = SwitchPort
    template_name = "network/switch-port-create.html"
    form_class = SwitchPortForm
    success_message = _(u'Successfully created switch port!')


682
class SwitchPortDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
683 684 685 686 687 688 689 690 691 692 693
    model = SwitchPort
    template_name = "network/confirm/base_delete.html"

    def get_success_url(self):
        next = self.request.POST.get('next')
        if next:
            return next
        else:
            return reverse_lazy('network.switch_port_list')


694
class VlanList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
695 696 697 698 699 700
    model = Vlan
    table_class = VlanTable
    template_name = "network/vlan-list.html"
    table_pagination = False


Kálmán Viktor committed
701 702 703 704
class VlanAclUpdateView(AclUpdateView):
    model = Vlan


705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725
class VlanMagicMixin(object):
    def _get_ajax(self, *args, **kwargs):
        GET = self.request.GET
        result = {}
        if "network4" in GET and "network6" in GET:
            try:
                result["ipv6_template"] = Vlan._magic_ipv6_template(
                    IPNetwork(GET['network4']),
                    IPNetwork(GET['network6']))
            except:
                result["ipv6_template"] = ""
        return JsonResponse({k: unicode(result[k] or "") for k in result})

    def get(self, *args, **kwargs):
        if self.request.is_ajax():
            return self._get_ajax(*args, **kwargs)
        else:
            return super(VlanMagicMixin, self).get(*args, **kwargs)


class VlanDetail(VlanMagicMixin, LoginRequiredMixin, SuperuserRequiredMixin,
726
                 SuccessMessageMixin, UpdateView):
727 728 729 730 731
    model = Vlan
    template_name = "network/vlan-edit.html"
    form_class = VlanForm
    slug_field = 'vid'
    slug_url_kwarg = 'vid'
732
    success_message = _(u'Succesfully modified vlan %(name)s!')
733 734 735

    def get_context_data(self, **kwargs):
        context = super(VlanDetail, self).get_context_data(**kwargs)
736
        context['host_list'] = SmallHostTable(self.object.host_set.all())
737
        context['dhcp_list'] = SmallDhcpTable(self.object.get_dhcp_clients())
738
        context['vlan_vid'] = self.kwargs.get('vid')
739 740
        context['acl'] = AclUpdateView.get_acl_data(
            self.object, self.request.user, 'network.vlan-acl')
741
        context['aclform'] = AclUserOrGroupAddForm()
742 743 744 745 746
        return context

    success_url = reverse_lazy('network.vlan_list')


747
class VlanCreate(VlanMagicMixin, LoginRequiredMixin, SuperuserRequiredMixin,
748
                 SuccessMessageMixin, InitialOwnerMixin, CreateView):
749 750 751
    model = Vlan
    template_name = "network/vlan-create.html"
    form_class = VlanForm
752
    success_message = _(u'Successfully created vlan %(name)s!')
753 754


755
class VlanDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772
    model = Vlan
    template_name = "network/confirm/base_delete.html"

    def get_success_url(self):
        next = self.request.POST.get('next')
        if next:
            return next
        else:
            return reverse_lazy('network.vlan_list')

    def get_object(self, queryset=None):
        """ we identify vlans by vid and not pk """
        return Vlan.objects.get(vid=self.kwargs['vid'])

    def delete(self, request, *args, **kwargs):
        self.object = self.get_object()
        if unicode(self.object) != request.POST.get('confirm'):
773
            messages.error(request, _(u"Object name does not match!"))
774 775 776
            return self.get(request, *args, **kwargs)

        response = super(VlanDelete, self).delete(request, *args, **kwargs)
777
        messages.success(request, _(u"Vlan successfully deleted!"))
778 779 780 781 782 783 784 785 786 787
        return response

    def get_context_data(self, **kwargs):
        context = super(VlanDelete, self).get_context_data(**kwargs)

        deps = []
        # hosts
        hosts = Host.objects.filter(vlan=self.object).all()
        if len(hosts) > 0:
            deps.append({
788
                'name': _('Hosts'),
789 790 791 792 793 794 795
                'data': hosts
            })

            # records
            records = Record.objects.filter(host__in=deps[0]['data'])
            if len(records) > 0:
                deps.append({
796
                    'name': _('Records'),
797 798 799 800 801 802 803 804
                    'data':  records
                })

        context['deps'] = deps
        context['confirmation'] = True
        return context


805 806
class VlanGroupList(LoginRequiredMixin, SuperuserRequiredMixin,
                    SingleTableView):
807 808 809 810 811 812
    model = VlanGroup
    table_class = VlanGroupTable
    template_name = "network/vlan-group-list.html"
    table_pagination = False


813 814
class VlanGroupDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                      SuccessMessageMixin, UpdateView):
815 816 817 818
    model = VlanGroup
    template_name = "network/vlan-group-edit.html"
    form_class = VlanGroupForm
    success_url = reverse_lazy('network.vlan_group_list')
819
    success_message = _(u'Successfully modified vlan group %(name)s!')
820 821 822 823 824 825 826

    def get_context_data(self, *args, **kwargs):
        context = super(VlanGroupDetail, self).get_context_data(**kwargs)
        context['vlangroup_pk'] = self.object.pk
        return context


827
class VlanGroupCreate(LoginRequiredMixin, SuperuserRequiredMixin,
828
                      SuccessMessageMixin, InitialOwnerMixin, CreateView):
829 830 831
    model = VlanGroup
    template_name = "network/vlan-group-create.html"
    form_class = VlanGroupForm
832
    success_message = _(u'Successfully created vlan group %(name)s!')
833 834


835
class VlanGroupDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862
    model = VlanGroup
    template_name = "network/confirm/base_delete.html"

    def get_success_url(self):
        next = self.request.POST.get('next')
        if next:
            return next
        else:
            return reverse_lazy('network.vlan_group_list')


def remove_host_group(request, **kwargs):
    host = Host.objects.get(pk=kwargs['pk'])
    group = Group.objects.get(pk=kwargs['group_pk'])

    # for get we show the confirmation page
    if request.method == "GET":
        return render(request,
                      'network/confirm/remove_host_group.html',
                      {
                          'group': group.name,
                          'host': host.hostname
                      })

    # for post we actually remove the group from the host
    elif request.method == "POST":
        host.groups.remove(group)
863
        if not request.is_ajax():
864
            messages.success(request, _(u"Successfully removed %(host)s from "
865 866 867 868
                                        "%(group)s group!" % {
                                            'host': host,
                                            'group': group
                                        }))
869 870 871 872 873 874 875 876 877 878
        return redirect(reverse_lazy('network.host',
                                     kwargs={'pk': kwargs['pk']}))


def add_host_group(request, **kwargs):
    group_pk = request.POST.get('group')
    if request.method == "POST" and group_pk:
        host = Host.objects.get(pk=kwargs['pk'])
        group = Group.objects.get(pk=group_pk)
        host.groups.add(group)
879
        if not request.is_ajax():
880
            messages.success(request, _(u"Successfully added %(host)s to group"
881 882 883 884
                                        " %(group)s!" % {
                                            'host': host,
                                            'group': group
                                        }))
885
        return redirect(reverse_lazy('network.host', kwargs=kwargs))
886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929


def remove_switch_port_device(request, **kwargs):
    device = EthernetDevice.objects.get(pk=kwargs['device_pk'])
    # for get we show the confirmation page
    if request.method == "GET":
        return render(request, 'network/confirm/base_delete.html',
                      {'object': device})

    # for post we actually remove the group from the host
    elif request.method == "POST":
        device.delete()
        if not request.is_ajax():
            messages.success(request, _(u"Successfully deleted ethernet device"
                                        " %(name)s!" % {
                                            'name': device.name,
                                        }))
        return redirect(reverse_lazy('network.switch_port',
                                     kwargs={'pk': kwargs['pk']}))


def add_switch_port_device(request, **kwargs):
    device_name = request.POST.get('device_name')

    if (request.method == "POST" and device_name and len(device_name) > 0
       and EthernetDevice.objects.filter(name=device_name).count() == 0):

        switch_port = SwitchPort.objects.get(pk=kwargs['pk'])
        new_device = EthernetDevice(name=device_name, switch_port=switch_port)
        new_device.save()
        if not request.is_ajax():
            messages.success(request, _(u"Successfully added %(name)s to this"
                                        " switch port" % {
                                            'name': device_name,
                                        }))
        return redirect(reverse_lazy('network.switch_port', kwargs=kwargs))

    elif not len(device_name) > 0:
        messages.error(request, _("Ethernet device name cannot be empty!"))
        return redirect(reverse_lazy('network.switch_port', kwargs=kwargs))
    elif EthernetDevice.objects.get(name=device_name) is not None:
        messages.error(request, _("There is already an ethernet device with"
                                  " that name!"))
        return redirect(reverse_lazy('network.switch_port', kwargs=kwargs))