views.py 32.7 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
from itertools import chain
Kálmán Viktor committed
46
from dashboard.views import AclUpdateView
47
from dashboard.forms import AclUserOrGroupAddForm
48

49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65
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)

66

67 68 69 70 71 72 73 74 75 76
class MagicMixin(object):

    def get(self, *args, **kwargs):
        if self.request.is_ajax():
            result = self._get_ajax(*args, **kwargs)
            return JsonResponse({k: unicode(result[k] or "") for k in result})
        else:
            return super(MagicMixin, self).get(*args, **kwargs)


77 78 79 80 81
class InitialOwnerMixin(FormMixin):
    def get_initial(self):
        initial = super(InitialOwnerMixin, self).get_initial()
        initial['owner'] = self.request.user
        return initial
82 83


84
class IndexView(LoginRequiredMixin, SuperuserRequiredMixin, TemplateView):
85 86 87 88 89 90
    template_name = "network/index.html"

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

        size = 13
91 92
        blacklists = BlacklistItem.objects.all().order_by(
            '-modified_at')[:size]
93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118
        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


119 120
class BlacklistList(LoginRequiredMixin, SuperuserRequiredMixin,
                    SingleTableView):
121 122
    model = BlacklistItem
    table_class = BlacklistItemTable
123 124 125 126
    template_name = "network/blacklist-list.html"
    table_pagination = False


127 128
class BlacklistDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                      SuccessMessageMixin, UpdateView):
129
    model = BlacklistItem
130
    template_name = "network/blacklist-edit.html"
131 132
    form_class = BlacklistItemForm
    success_message = _(u'Successfully modified blacklist item'
133 134 135 136 137 138 139 140 141 142 143 144
                        '%(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


145 146
class BlacklistCreate(LoginRequiredMixin, SuperuserRequiredMixin,
                      SuccessMessageMixin, CreateView):
147
    model = BlacklistItem
148
    template_name = "network/blacklist-create.html"
149 150
    form_class = BlacklistItemForm
    success_message = _(u'Successfully created blacklist item '
151 152 153
                        '%(ipv4)s - %(type)s!')


154
class BlacklistDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
155
    model = BlacklistItem
156 157 158 159 160 161
    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:
162
            to_delete = BlacklistItem.objects.get(pk=self.kwargs['pk'])
163 164 165 166 167 168 169 170 171 172 173 174 175
            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')


176
class DomainList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
177 178 179 180 181 182
    model = Domain
    table_class = DomainTable
    template_name = "network/domain-list.html"
    table_pagination = False


183 184
class DomainDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                   SuccessMessageMixin, UpdateView):
185 186 187
    model = Domain
    template_name = "network/domain-edit.html"
    form_class = DomainForm
188
    success_message = _(u'Successfully modified domain %(name)s!')
189 190 191 192 193 194 195

    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)
196 197 198 199 200 201 202
        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(
203
                    instance__destroyed_at=None)
204 205
            )
        )
206 207 208 209
        context['record_list'] = SmallRecordTable(q)
        return context


210
class DomainCreate(LoginRequiredMixin, SuperuserRequiredMixin,
211
                   SuccessMessageMixin, InitialOwnerMixin, CreateView):
212 213 214
    model = Domain
    template_name = "network/domain-create.html"
    form_class = DomainForm
215
    success_message = _(u'Successfully created domain %(name)s!')
216 217


218
class DomainDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
219 220 221 222 223 224 225 226 227 228 229 230 231
    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'):
232
            messages.error(request, _(u"Object name does not match!"))
233 234 235
            return self.get(request, *args, **kwargs)

        response = super(DomainDelete, self).delete(request, *args, **kwargs)
236
        messages.success(request, _(u"Domain successfully deleted!"))
237 238 239 240 241
        return response

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

242
        records_from_hosts = _(u'Records from hosts')
243 244 245 246 247
        deps = []
        # vlans
        vlans = Vlan.objects.filter(domain=self.object).all()
        if len(vlans) > 0:
            deps.append({
248
                'name': _('Vlans'),
249 250 251 252 253 254 255
                'data': vlans
            })

            # hosts
            hosts = Host.objects.filter(vlan__in=deps[0]['data'])
            if len(hosts) > 0:
                deps.append({
256
                    'name': _('Hosts'),
257 258 259 260 261
                    'data':  hosts
                })

                # records
                records = Record.objects.filter(
262 263
                    host__in=deps[1]['data']
                    # Q(domain=self.object) | (host__in=deps[1]['data'])
264 265 266
                )
                if len(records) > 0:
                    deps.append({
267
                        'name': records_from_hosts,
268 269 270
                        'data': records
                    })

271 272 273 274
        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)
275
            n = indexes.index(records_from_hosts) if len(indexes) > 0 else 0
276
            deps.append({
277
                'name': u'Records only from the domain',
278 279 280 281
                'data': records.exclude(pk__in=deps[n]['data']) if n > 0
                else records
            })

282 283 284 285 286
        context['deps'] = deps
        context['confirmation'] = True
        return context


287
class GroupList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
288 289 290 291 292 293
    model = Group
    table_class = GroupTable
    template_name = "network/group-list.html"
    table_pagination = False


294
class GroupCreate(LoginRequiredMixin, SuperuserRequiredMixin,
295
                  SuccessMessageMixin, InitialOwnerMixin, CreateView):
296 297 298
    model = Group
    template_name = "network/group-create.html"
    form_class = GroupForm
299
    success_message = _(u'Successfully created host group %(name)s!')
300 301


302 303
class GroupDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                  SuccessMessageMixin, UpdateView):
304 305 306
    model = Group
    template_name = "network/group-edit.html"
    form_class = GroupForm
307
    success_message = _(u'Successfully modified host group %(name)s!')
308 309 310 311 312

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

313 314 315
    def get_context_data(self, *args, **kwargs):
        context = super(GroupDetail, self).get_context_data(**kwargs)

316 317
        context['group_pk'] = self.object.pk

318 319 320 321 322
        # records
        q = Rule.objects.filter(hostgroup=self.object)
        context['rule_list'] = SmallRuleTable(q)
        return context

323

324
class GroupDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340
    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


341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359
class HostMagicMixin(MagicMixin):
    def _get_ajax(self, *args, **kwargs):
        GET = self.request.GET
        result = {}
        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"]) or ""
            except:
                result["ipv6"] = ""
        else:
            try:
                result.update(vlan.get_new_address())
            except ValidationError:
                result["ipv4"] = ""
                result["ipv6"] = ""
        return result


360
class HostList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
361 362 363 364 365 366 367 368 369 370 371 372 373 374
    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:
375
            data = Host.objects.filter(vlan=vlan_id).select_related()
376
        else:
377
            data = Host.objects.select_related()
378 379 380
        return data


381
class HostDetail(HostMagicMixin, LoginRequiredMixin, SuperuserRequiredMixin,
382
                 SuccessMessageMixin, UpdateView):
383 384 385
    model = Host
    template_name = "network/host-edit.html"
    form_class = HostForm
386
    success_message = _(u'Successfully modified host %(hostname)s!')
387

388 389
    def _get_ajax(self, *args, **kwargs):
        if "vlan" not in self.request.GET:
390 391 392
            host = Host.objects.get(pk=kwargs['pk'])
            host = {
                'hostname': host.hostname,
393 394
                'ipv4': str(host.ipv4),
                'ipv6': str(host.ipv6),
395 396
                'fqdn': host.get_fqdn()
            }
397
            return host
398
        else:
399 400 401 402 403
            return super(HostDetail, self)._get_ajax(*args, **kwargs)

    def get(self, request, *args, **kwargs):
        self.object = self.get_object()
        return super(HostDetail, self).get(request, *args, **kwargs)
404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440

    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']

441 442 443
        from network.tables import HostRecordsTable
        context['records_table'] = HostRecordsTable(
            Record.objects.filter(host=self.get_object()),
444
            request=self.request, template="django_tables2/table_no_page.html"
445 446
        )

447 448 449 450 451 452 453
        return context

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


454
class HostCreate(HostMagicMixin, LoginRequiredMixin, SuperuserRequiredMixin,
455
                 SuccessMessageMixin, InitialOwnerMixin, CreateView):
456 457 458
    model = Host
    template_name = "network/host-create.html"
    form_class = HostForm
459
    success_message = _(u'Successfully created host %(hostname)s!')
460

461 462 463 464 465 466 467
    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:
468 469
            if not initial['vlan'].isnumeric():
                raise Http404()
470 471 472 473 474 475 476
            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

477

478
class HostDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495
    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({
496
                'name': _('Records'),
497 498 499 500 501 502 503 504 505 506
                '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'):
507
            messages.error(request, _(u"Object name does not match!"))
508 509 510
            return self.get(request, *args, **kwargs)

        response = super(HostDelete, self).delete(request, *args, **kwargs)
511
        messages.success(request, _(u"Host successfully deleted!"))
512 513 514
        return response


515
class RecordList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
516 517 518 519 520
    model = Record
    table_class = RecordTable
    template_name = "network/record-list.html"
    table_pagination = False

521 522 523 524 525 526 527 528 529 530 531 532 533
    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

534

535 536
class RecordDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                   SuccessMessageMixin, UpdateView):
537 538 539 540
    model = Record
    template_name = "network/record-edit.html"
    form_class = RecordForm
    # TODO fqdn
541
    success_message = _(u'Successfully modified record!')
542 543 544 545 546 547 548 549 550 551 552 553

    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)


554
class RecordCreate(LoginRequiredMixin, SuperuserRequiredMixin,
555
                   SuccessMessageMixin, InitialOwnerMixin, CreateView):
556 557 558 559
    model = Record
    template_name = "network/record-create.html"
    form_class = RecordForm
    # TODO fqdn
560
    success_message = _(u'Successfully created record!')
561

562
    def get_initial(self):
563 564 565
        initial = super(RecordCreate, self).get_initial()
        initial['domain'] = self.request.GET.get('domain')

566 567 568 569 570 571 572 573 574 575 576 577 578
        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(),
            })

579
        return initial
580

581

582
class RecordDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
583 584 585 586 587 588 589 590 591 592 593
    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')


594
class RuleList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
595 596 597 598 599
    model = Rule
    table_class = RuleTable
    template_name = "network/rule-list.html"
    table_pagination = False

600 601 602 603 604
    def get_table_data(self):
        return Rule.objects.select_related('host', 'hostgroup', 'vlan',
                                           'vlangroup', 'firewall',
                                           'foreign_network', 'owner')

605

606 607
class RuleDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                 SuccessMessageMixin, UpdateView):
608 609 610
    model = Rule
    template_name = "network/rule-edit.html"
    form_class = RuleForm
611
    success_message = _(u'Successfully modified rule!')
612 613 614 615 616 617 618 619 620 621 622 623 624 625

    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


626
class RuleCreate(LoginRequiredMixin, SuperuserRequiredMixin,
627
                 SuccessMessageMixin, InitialOwnerMixin, CreateView):
628 629 630
    model = Rule
    template_name = "network/rule-create.html"
    form_class = RuleForm
631
    success_message = _(u'Successfully created rule!')
632

633
    def get_initial(self):
634 635
        initial = super(RuleCreate, self).get_initial()
        initial.update({
636 637
            'host': self.request.GET.get('host'),
            'hostgroup': self.request.GET.get('hostgroup')
638 639
        })
        return initial
640

641

642
class RuleDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
643 644 645 646 647 648 649 650 651 652 653
    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')


654 655
class SwitchPortList(LoginRequiredMixin, SuperuserRequiredMixin,
                     SingleTableView):
656 657 658 659 660 661
    model = SwitchPort
    table_class = SwitchPortTable
    template_name = "network/switch-port-list.html"
    table_pagination = False


662 663
class SwitchPortDetail(LoginRequiredMixin, SuperuserRequiredMixin,
                       SuccessMessageMixin, UpdateView):
664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680
    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


681 682
class SwitchPortCreate(LoginRequiredMixin, SuperuserRequiredMixin,
                       SuccessMessageMixin, CreateView):
683 684 685 686 687 688
    model = SwitchPort
    template_name = "network/switch-port-create.html"
    form_class = SwitchPortForm
    success_message = _(u'Successfully created switch port!')


689
class SwitchPortDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
690 691 692 693 694 695 696 697 698 699 700
    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')


701
class VlanList(LoginRequiredMixin, SuperuserRequiredMixin, SingleTableView):
702 703 704 705 706 707
    model = Vlan
    table_class = VlanTable
    template_name = "network/vlan-list.html"
    table_pagination = False


Kálmán Viktor committed
708 709 710 711
class VlanAclUpdateView(AclUpdateView):
    model = Vlan


712
class VlanMagicMixin(MagicMixin):
713 714 715 716 717
    def _get_ajax(self, *args, **kwargs):
        GET = self.request.GET
        result = {}
        if "network4" in GET and "network6" in GET:
            try:
718 719 720
                result["ipv6_template"], result["host_ipv6_prefixlen"] = (
                    Vlan._magic_ipv6_template(IPNetwork(GET['network4']),
                                              IPNetwork(GET['network6'])))
721
            except:
722
                result["ipv6_template"] = result["host_ipv6_prefixlen"] = ""
723
        return result
724 725 726


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

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

    success_url = reverse_lazy('network.vlan_list')


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


756
class VlanDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773
    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'):
774
            messages.error(request, _(u"Object name does not match!"))
775 776 777
            return self.get(request, *args, **kwargs)

        response = super(VlanDelete, self).delete(request, *args, **kwargs)
778
        messages.success(request, _(u"Vlan successfully deleted!"))
779 780 781 782 783 784 785 786 787 788
        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({
789
                'name': _('Hosts'),
790 791 792 793 794 795 796
                'data': hosts
            })

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

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


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


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

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


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


836
class VlanGroupDelete(LoginRequiredMixin, SuperuserRequiredMixin, DeleteView):
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 863
    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)
864
        if not request.is_ajax():
865
            messages.success(request, _(u"Successfully removed %(host)s from "
866 867 868 869
                                        "%(group)s group!" % {
                                            'host': host,
                                            'group': group
                                        }))
870 871 872 873 874 875 876 877 878 879
        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)
880
        if not request.is_ajax():
881
            messages.success(request, _(u"Successfully added %(host)s to group"
882 883 884 885
                                        " %(group)s!" % {
                                            'host': host,
                                            'group': group
                                        }))
886
        return redirect(reverse_lazy('network.host', kwargs=kwargs))
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 930


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))