PKt[CR;%%iprestrict/__init__.pyfrom restrictor import IPRestrictor PKH1H>T T iprestrict/admin.pyfrom iprestrict import models from django.contrib import admin from django import forms import ip_utils as ipu class RuleAdmin(admin.ModelAdmin): model = models.Rule exclude = ('rank',) list_display = ('url_pattern', 'ip_group', 'is_allowed', 'move_up_url', 'move_down_url') class IPRangeForm(forms.ModelForm): class Meta: model = models.IPRange def clean_cidr_prefix_length(self): cidr = self.cleaned_data['cidr_prefix_length'] if cidr: if not (1 <= cidr <= 31): raise forms.ValidationError("Must be a number between 1 and 31") return cidr def clean(self): cleaned_data = super(IPRangeForm, self).clean() first_ip = cleaned_data.get('first_ip') if first_ip is None: # first_ip is Mandatory, so just let the default validator catch this return cleaned_data last_ip = cleaned_data['last_ip'] cidr = cleaned_data['cidr_prefix_length'] if last_ip and cidr: raise forms.ValidationError("Don't specify the Last IP if you specified a CIDR prefix length") if last_ip: if ipu.get_version(first_ip) != ipu.get_version(last_ip): raise forms.ValidationError("Last IP should be the same type as First IP (%s)" % ipu.get_version(first_ip)) if ipu.get_version(last_ip) != 'ipv6': # Ignore rest of validation for ipv6, support isn't there yet if ipu.to_number(first_ip) > ipu.to_number(last_ip): raise forms.ValidationError("Last IP should be greater than First IP") if cidr: # With CIDR the starting address could be different than the one # the user specified. Making sure it is set to the first ip in the # subnet. start, end = ipu.cidr_to_range(first_ip, cidr) cleaned_data['first_ip'] = ipu.to_ip(start) return cleaned_data class IPRangeInline(admin.TabularInline): model = models.IPRange form = IPRangeForm fields = ['first_ip', 'cidr_prefix_length', 'last_ip', 'ip_type'] readonly_fields = ['ip_type'] extra = 2 class IPGroupAdmin(admin.ModelAdmin): model = models.IPGroup inlines = [IPRangeInline] admin.site.register(models.Rule, RuleAdmin) admin.site.register(models.IPGroup, IPGroupAdmin) PKH1HWwiprestrict/decorators.pyfrom functools import wraps from django.utils.translation import ugettext as _ from django.contrib.admin.forms import AdminAuthenticationForm from django.contrib.auth.views import login from django.contrib.auth import REDIRECT_FIELD_NAME from django.http import HttpResponseForbidden # Copied from django staff_member_required # Why isn't this provided by Django? def superuser_required(view_func): """ Decorator for views that checks that the user is logged in and is a staff member, displaying the login page if necessary. """ @wraps(view_func) def _checklogin(request, *args, **kwargs): if request.user.is_active and request.user.is_superuser: # The user is valid. Continue to the admin page. return view_func(request, *args, **kwargs) if request.user.is_authenticated(): return HttpResponseForbidden('Forbidden!') assert hasattr(request, 'session'), "The Django admin requires session middleware to be installed. Edit your MIDDLEWARE_CLASSES setting to insert 'django.contrib.sessions.middleware.SessionMiddleware'." defaults = { 'template_name': 'admin/login.html', 'authentication_form': AdminAuthenticationForm, 'extra_context': { 'title': _('Log in'), 'app_path': request.get_full_path(), REDIRECT_FIELD_NAME: request.get_full_path(), }, } return login(request, **defaults) return _checklogin PKH1HB6iprestrict/ip_utils.py def get_version(ip): return 'ipv4' if '.' in ip else 'ipv6' def to_number(ip): parts= ip.split('.') parts = [int(p) for p in reversed(parts)] nr = 0 for i, d in enumerate(parts): nr += (256 ** i) * d return nr def to_ip(number): mask = int('1' * 8, 2) parts = [] for i in range(4): shifted_number = number >> (8 * i) parts.append(shifted_number & mask) return '.'.join(map(lambda i: str(i), reversed(parts))) def cidr_to_range(ip, prefix_length): ip = to_number(ip) start_mask = int('1' * prefix_length, 2) << (32 - prefix_length) end_mask = int('1' * (32 - prefix_length), 2) start = ip & start_mask end = start | end_mask return (start, end) PKH1HB!!iprestrict/middleware.pyfrom django.core import exceptions from django.utils import log from iprestrict.models import ReloadRulesRequest from iprestrict import IPRestrictor from django.conf import settings logger = log.getLogger(__name__) class IPRestrictMiddleware(object): def __init__(self): self.restrictor = IPRestrictor() self.trusted_proxies = getattr(settings, 'TRUSTED_PROXIES', tuple()) self.dont_reload_rules = getattr(settings, 'DONT_RELOAD_RULES', False) def process_request(self, request): if not self.dont_reload_rules: self.reload_rules_if_needed() url = request.path_info client_ip = self.extract_client_ip(request) if self.restrictor.is_restricted(url, client_ip): logger.info("Denying access of %s to %s" % (url, client_ip)) raise exceptions.PermissionDenied def extract_client_ip(self, request): client_ip = request.META['REMOTE_ADDR'] forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR') if forwarded_for is not None: forwarded_for = [ip.strip() for ip in forwarded_for.split(',')] closest_proxy = client_ip client_ip = forwarded_for.pop(0) proxies = [closest_proxy] + forwarded_for for proxy in proxies: if proxy not in self.trusted_proxies: logger.info("Client IP %s forwarded by untrusted proxy %s" % (client_ip, proxy)) raise exceptions.PermissionDenied return client_ip def reload_rules_if_needed(self): last_reload_request = ReloadRulesRequest.last_request() if last_reload_request is not None: if self.restrictor.last_reload < last_reload_request: self.restrictor.reload_rules() PKH1HܗCCiprestrict/models.pyfrom django.core.urlresolvers import reverse from django.db import models from datetime import datetime import re from iprestrict import ip_utils as ipu class IPGroup(models.Model): class Meta: verbose_name = "IP Group" name = models.CharField(max_length=100) description = models.TextField(null=True, blank=True) def __init__(self, *args, **kwargs): models.Model.__init__(self, *args, **kwargs) self.load_ranges() def load_ranges(self): self._ranges = { 'ipv4': [], 'ipv6': [] } for r in self.iprange_set.all(): self._ranges[r.ip_type].append(r) def ranges(self, ip_type='ipv4'): return self._ranges[ip_type] def matches(self, ip): ip_type = ipu.get_version(ip) for r in self.ranges(ip_type): if ip in r: return True return False def ranges_str(self): return ', '.join([str(r) for r in self.ranges()]) def __unicode__(self): return self.name class IPRange(models.Model): class Meta: verbose_name = "IP Range" ip_group = models.ForeignKey(IPGroup) first_ip = models.GenericIPAddressField() cidr_prefix_length = models.PositiveSmallIntegerField(null=True, blank=True) last_ip = models.GenericIPAddressField(null=True, blank=True) @property def start(self): if self.cidr_prefix_length is not None: start, end = ipu.cidr_to_range(self.first_ip, self.cidr_prefix_length) return start else: return ipu.to_number(self.first_ip) @property def end(self): if self.last_ip is not None: return ipu.to_number(self.last_ip) if self.cidr_prefix_length is not None: start, end = ipu.cidr_to_range(self.first_ip, self.cidr_prefix_length) return end return self.start @property def ip_type(self): if not self.first_ip: return '' return ipu.get_version(self.first_ip) def __contains__(self, ip): ip_nr = ipu.to_number(ip) return self.start <= ip_nr <= self.end def __unicode__(self): result = str(self.first_ip) if self.cidr_prefix_length is not None: result += '/' + str(self.cidr_prefix_length) elif self.last_ip is not None: result += '-' + str(self.last_ip) return result class Rule(models.Model): class Meta: ordering = ['rank', 'id'] ACTION_CHOICES = ( ('A', 'ALLOW'), ('D', 'DENY') ) url_pattern = models.CharField(max_length=500) ip_group = models.ForeignKey(IPGroup, default=1) action = models.CharField(max_length=1, choices=ACTION_CHOICES, default='D') rank = models.IntegerField(blank=True) @property def regex(self): if not hasattr(self, '_regex'): self._regex = re.compile(self.url_pattern) return self._regex def matches_url(self, url): if self.url_pattern == 'ALL': return True else: return (self.regex.match(url) is not None) def matches_ip(self, ip): return self.ip_group.matches(ip) def is_restricted(self): return self.action != 'A' def is_allowed(self): return self.action == 'A' is_allowed.boolean = True is_allowed.short_description = 'Is allowed?' def action_str(self): return 'Allowed' if self.is_allowed() else 'Denied' def swap_with_rule(self, other): other.rank, self.rank = self.rank, other.rank other.save() self.save() def move_up(self): rules_above = Rule.objects.filter(rank__lt = self.rank).order_by('-rank') if len(rules_above) == 0: return self.swap_with_rule(rules_above[0]) def move_up_url(self): url = reverse('iprestrict.views.move_rule_up', args=[self.pk]) return 'Move Up' % url move_up_url.allow_tags = True move_up_url.short_description = 'Move Up' def move_down_url(self): url = reverse('iprestrict.views.move_rule_down', args=[self.pk]) return 'Move Down' % url move_down_url.allow_tags = True move_down_url.short_description = 'Move Down' def move_down(self): rules_below = Rule.objects.filter(rank__gt = self.rank) if len(rules_below) == 0: return self.swap_with_rule(rules_below[0]) def save(self, *args, **kwargs): if self.rank is None: max_aggr = Rule.objects.filter(rank__lt = 65000).aggregate(models.Max('rank')) max_rank = max_aggr.get('rank__max') if max_rank is None: max_rank = 0 self.rank = max_rank + 1 super(Rule, self).save(*args, **kwargs) class ReloadRulesRequest(models.Model): at = models.DateTimeField(auto_now_add=True) @classmethod def request_reload(cls): rrs = ReloadRulesRequest.objects.all() if len(rrs) > 0: obj = rrs[0] obj.at = datetime.now() obj.save() else: cls.objects.create() @staticmethod def last_request(): result = None rrs = ReloadRulesRequest.objects.all() if len(rrs) > 0: result = rrs[0].at return result PKH1Hl88iprestrict/restrictor.pyfrom iprestrict import models from datetime import datetime class IPRestrictor(object): def __init__(self): self.load_rules() def is_restricted(self, url, ip): for rule in self.rules: if rule.matches_url(url) and rule.matches_ip(ip): return rule.is_restricted() return False def load_rules(self): # We are caching the rules, to avoid DB lookup on each request self.rules = [r for r in models.Rule.objects.all()] self.last_reload = datetime.now() reload_rules = load_rules PKH1H))iprestrict/test_settings.pyfrom os import path # Django settings for mysite project. DEBUG = True TEMPLATE_DEBUG = DEBUG ADMINS = ( # ('Your Name', 'your_email@example.com'), ) MANAGERS = ADMINS DATABASES = { 'default': { 'ENGINE': 'django.db.backends.sqlite3', # Add 'postgresql_psycopg2', 'mysql', 'sqlite3' or 'oracle'. 'NAME': 'db.sqlite3', # Or path to database file if using sqlite3. 'USER': '', # Not used with sqlite3. 'PASSWORD': '', # Not used with sqlite3. 'HOST': '', # Set to empty string for localhost. Not used with sqlite3. 'PORT': '', # Set to empty string for default. Not used with sqlite3. } } # Local time zone for this installation. Choices can be found here: # http://en.wikipedia.org/wiki/List_of_tz_zones_by_name # although not all choices may be available on all operating systems. # On Unix systems, a value of None will cause Django to use the same # timezone as the operating system. # If running in a Windows environment this must be set to the same as your # system time zone. TIME_ZONE = 'Australia/Perth' # Language code for this installation. All choices can be found here: # http://www.i18nguy.com/unicode/language-identifiers.html LANGUAGE_CODE = 'en-us' SITE_ID = 1 # If you set this to False, Django will make some optimizations so as not # to load the internationalization machinery. USE_I18N = True # If you set this to False, Django will not format dates, numbers and # calendars according to the current locale. USE_L10N = True # If you set this to False, Django will not use timezone-aware datetimes. USE_TZ = False # Absolute filesystem path to the directory that will hold user-uploaded files. # Example: "/home/media/media.lawrence.com/media/" MEDIA_ROOT = '' # URL that handles the media served from MEDIA_ROOT. Make sure to use a # trailing slash. # Examples: "http://media.lawrence.com/media/", "http://example.com/media/" MEDIA_URL = '' # Absolute path to the directory static files should be collected to. # Don't put anything in this directory yourself; store your static files # in apps' "static/" subdirectories and in STATICFILES_DIRS. # Example: "/home/media/media.lawrence.com/static/" STATIC_ROOT = '' # URL prefix for static files. # Example: "http://media.lawrence.com/static/" STATIC_URL = '/static/' # Additional locations of static files STATICFILES_DIRS = ( # Put strings here, like "/home/html/static" or "C:/www/django/static". # Always use forward slashes, even on Windows. # Don't forget to use absolute paths, not relative paths. ) # List of finder classes that know how to find static files in # various locations. STATICFILES_FINDERS = ( 'django.contrib.staticfiles.finders.FileSystemFinder', 'django.contrib.staticfiles.finders.AppDirectoriesFinder', # 'django.contrib.staticfiles.finders.DefaultStorageFinder', ) # Make this unique, and don't share it with anybody. SECRET_KEY = 're)-uyql-#_jc4q&m99upnmn#epel3#1*lvkh9e^y=r9k0-6z%' # List of callables that know how to import templates from various sources. TEMPLATE_LOADERS = ( 'django.template.loaders.filesystem.Loader', 'django.template.loaders.app_directories.Loader', # 'django.template.loaders.eggs.Loader', ) MIDDLEWARE_CLASSES = ( 'django.middleware.common.CommonMiddleware', 'iprestrict.middleware.IPRestrictMiddleware', 'django.contrib.sessions.middleware.SessionMiddleware', 'django.middleware.csrf.CsrfViewMiddleware', 'django.contrib.auth.middleware.AuthenticationMiddleware', 'django.contrib.messages.middleware.MessageMiddleware', # Uncomment the next line for simple clickjacking protection: # 'django.middleware.clickjacking.XFrameOptionsMiddleware', ) ROOT_URLCONF = 'iprestrict.test_urls' # Python dotted path to the WSGI application used by Django's runserver. #WSGI_APPLICATION = 'mysite.wsgi.application' TEMPLATE_DIRS = ( # Put strings here, like "/home/html/django_templates" or "C:/www/django/templates". # Always use forward slashes, even on Windows. # Don't forget to use absolute paths, not relative paths. ) INSTALLED_APPS = ( 'django.contrib.auth', 'django.contrib.contenttypes', 'django.contrib.sessions', 'django.contrib.sites', 'django.contrib.messages', 'django.contrib.staticfiles', # Uncomment the next line to enable the admin: 'django.contrib.admin', # Uncomment the next line to enable admin documentation: # 'django.contrib.admindocs', 'south', 'templatetag_handlebars', 'iprestrict', ) # A sample logging configuration. The only tangible logging # performed by this configuration is to send an email to # the site admins on every HTTP 500 error when DEBUG=False. # See http://docs.djangoproject.com/en/dev/topics/logging for # more details on how to customize your logging configuration. LOGGING = { 'version': 1, 'disable_existing_loggers': False, 'formatters': { 'simple': { 'format': '%(levelname)s %(message)s' }, }, 'filters': { 'require_debug_false': { '()': 'django.utils.log.RequireDebugFalse' } }, 'handlers': { 'mail_admins': { 'level': 'ERROR', 'filters': ['require_debug_false'], 'class': 'django.utils.log.AdminEmailHandler' }, 'console': { 'level':'DEBUG', 'class':'logging.StreamHandler', 'formatter': 'simple' }, }, 'loggers': { 'django.request': { 'handlers': ['mail_admins'], 'level': 'ERROR', 'propagate': True, }, 'iprestrict': { 'handlers': ['console'], 'level': 'DEBUG', 'propagate': True, } } } TEST_RUNNER = 'discover_runner.DiscoverRunner' PKH1HU!iprestrict/test_settings_1_3_1.pyfrom os import path # Django settings for mysite project. DEBUG = True TEMPLATE_DEBUG = DEBUG ADMINS = ( # ('Your Name', 'your_email@example.com'), ) MANAGERS = ADMINS DATABASES = { 'default': { 'ENGINE': 'django.db.backends.sqlite3', # Add 'postgresql_psycopg2', 'mysql', 'sqlite3' or 'oracle'. 'NAME': 'db.sqlite3', 'USER': '', # Not used with sqlite3. 'PASSWORD': '', # Not used with sqlite3. 'HOST': '', # Set to empty string for localhost. Not used with sqlite3. 'PORT': '', # Set to empty string for default. Not used with sqlite3. } } # Local time zone for this installation. Choices can be found here: # http://en.wikipedia.org/wiki/List_of_tz_zones_by_name # although not all choices may be available on all operating systems. # On Unix systems, a value of None will cause Django to use the same # timezone as the operating system. # If running in a Windows environment this must be set to the same as your # system time zone. TIME_ZONE = 'Australia/Perth' # Language code for this installation. All choices can be found here: # http://www.i18nguy.com/unicode/language-identifiers.html LANGUAGE_CODE = 'en-us' SITE_ID = 1 # If you set this to False, Django will make some optimizations so as not # to load the internationalization machinery. USE_I18N = True # If you set this to False, Django will not format dates, numbers and # calendars according to the current locale USE_L10N = True # Absolute filesystem path to the directory that will hold user-uploaded files. # Example: "/home/media/media.lawrence.com/media/" MEDIA_ROOT = '' # URL that handles the media served from MEDIA_ROOT. Make sure to use a # trailing slash. # Examples: "http://media.lawrence.com/media/", "http://example.com/media/" MEDIA_URL = '' # Absolute path to the directory static files should be collected to. # Don't put anything in this directory yourself; store your static files # in apps' "static/" subdirectories and in STATICFILES_DIRS. # Example: "/home/media/media.lawrence.com/static/" STATIC_ROOT = '' # URL prefix for static files. # Example: "http://media.lawrence.com/static/" STATIC_URL = '/static/' # URL prefix for admin static files -- CSS, JavaScript and images. # Make sure to use a trailing slash. # Examples: "http://foo.com/static/admin/", "/static/admin/". ADMIN_MEDIA_PREFIX = '/static/admin/' # Additional locations of static files STATICFILES_DIRS = ( # Put strings here, like "/home/html/static" or "C:/www/django/static". # Always use forward slashes, even on Windows. # Don't forget to use absolute paths, not relative paths. ) # List of finder classes that know how to find static files in # various locations. STATICFILES_FINDERS = ( 'django.contrib.staticfiles.finders.FileSystemFinder', 'django.contrib.staticfiles.finders.AppDirectoriesFinder', # 'django.contrib.staticfiles.finders.DefaultStorageFinder', ) # Make this unique, and don't share it with anybody. SECRET_KEY = '' # List of callables that know how to import templates from various sources. TEMPLATE_LOADERS = ( 'django.template.loaders.filesystem.Loader', 'django.template.loaders.app_directories.Loader', # 'django.template.loaders.eggs.Loader', ) MIDDLEWARE_CLASSES = ( 'django.middleware.common.CommonMiddleware', 'iprestrict.middleware.IPRestrictMiddleware', 'django.contrib.sessions.middleware.SessionMiddleware', 'django.middleware.csrf.CsrfViewMiddleware', 'django.contrib.auth.middleware.AuthenticationMiddleware', 'django.contrib.messages.middleware.MessageMiddleware', ) ROOT_URLCONF = 'iprestrict.urls' TEMPLATE_DIRS = ( # Put strings here, like "/home/html/django_templates" or "C:/www/django/templates". # Always use forward slashes, even on Windows. # Don't forget to use absolute paths, not relative paths. path.join(path.dirname(path.dirname(__file__)), 'templates') ) INSTALLED_APPS = ( 'django.contrib.auth', 'django.contrib.contenttypes', 'django.contrib.sessions', 'django.contrib.sites', 'django.contrib.messages', 'django.contrib.staticfiles', # Uncomment the next line to enable the admin: # 'django.contrib.admin', # Uncomment the next line to enable admin documentation: # 'django.contrib.admindocs', 'iprestrict', ) # A sample logging configuration. The only tangible logging # performed by this configuration is to send an email to # the site admins on every HTTP 500 error. # See http://docs.djangoproject.com/en/dev/topics/logging for # more details on how to customize your logging configuration. LOGGING = { 'version': 1, 'disable_existing_loggers': False, 'handlers': { 'mail_admins': { 'level': 'ERROR', 'class': 'django.utils.log.AdminEmailHandler' } }, 'loggers': { 'django.request': { 'handlers': ['mail_admins'], 'level': 'ERROR', 'propagate': True, }, } } TEST_RUNNER = 'discover_runner.DiscoverRunner' PKH1H" iprestrict/test_urls.py# This file is to be used for testing only from django.conf.urls import patterns, include, url from django.contrib.staticfiles.urls import staticfiles_urlpatterns from django.contrib import admin admin.autodiscover() urlpatterns = patterns('iprestrict.views', url(r'^iprestrict/$', 'test_rules_page'), url(r'^iprestrict/move_rule_up/(?P\d+)[/]?$', 'move_rule_up'), url(r'^iprestrict/move_rule_down/(?P\d+)[/]?$', 'move_rule_down'), url(r'^iprestrict/reload_rules[/]?$', 'reload_rules'), url(r'^iprestrict/test_match[/]?$', 'test_match'), ) urlpatterns += patterns('', url(r'^admin/', include(admin.site.urls)), ) urlpatterns += staticfiles_urlpatterns() PKH1Hiiiprestrict/urls.pyfrom django.conf.urls import patterns, include, url urlpatterns = patterns('iprestrict.views', url(r'^$', 'test_rules_page'), url(r'^move_rule_up/(?P\d+)[/]?$', 'move_rule_up'), url(r'^move_rule_down/(?P\d+)[/]?$', 'move_rule_down'), url(r'^reload_rules[/]?$', 'reload_rules'), url(r'^test_match[/]?$', 'test_match'), ) PKH1H3  iprestrict/views.pyfrom django.core.urlresolvers import reverse from django.http import HttpResponse, HttpResponseRedirect from iprestrict import models from iprestrict.restrictor import IPRestrictor from iprestrict.decorators import superuser_required from django.shortcuts import render_to_response import json @superuser_required def move_rule_up(request, rule_id): rule = models.Rule.objects.get(pk=rule_id) rule.move_up() return HttpResponseRedirect(reverse('admin:iprestrict_rule_changelist')) @superuser_required def move_rule_down(request, rule_id): rule = models.Rule.objects.get(pk=rule_id) rule.move_down() return HttpResponseRedirect(reverse('admin:iprestrict_rule_changelist')) @superuser_required def reload_rules(request): models.ReloadRulesRequest.request_reload() return HttpResponse('ok') @superuser_required def test_rules_page(request): return render_to_response('iprestrict/test_rules.html') @superuser_required def test_match(request): url = request.REQUEST['url'] ip = request.REQUEST['ip'] matching_rule_id, action = find_matching_rule(url, ip) rules = list_rules(matching_rule_id, url, ip) if matching_rule_id is None: result = { 'action': 'Allowed', 'msg': 'No rules matched.', } else: result = { 'action': action, 'msg': 'URL matched Rule highlighted below.' } result['rules'] = rules return HttpResponse(json.dumps(result)) def find_matching_rule(url, ip): for r in models.Rule.objects.all(): if r.matches_url(url) and r.matches_ip(ip): return (r.pk, r.action_str()) return (None, None) def list_rules(matching_rule_id, url, ip): return [map_rule(r, matching_rule_id, url, ip) for r in models.Rule.objects.all()] def map_rule(r, matching_rule_id, url, ip): rule = { 'url_pattern': { 'value': r.url_pattern, 'matchStatus': 'match' if r.matches_url(url) else 'noMatch' }, 'ip_group': { 'name': r.ip_group.name, 'ranges': r.ip_group.ranges_str(), 'matchStatus': 'match' if r.matches_ip(ip) else 'noMatch' }, 'action': r.action_str(), } if r.pk == matching_rule_id: rule['matched'] = True return rule PKt[C!iprestrict/management/__init__.pyPKt[C*iprestrict/management/commands/__init__.pyPKt[CE[''-iprestrict/management/commands/importrules.pyfrom django.core.management.base import BaseCommand from django.core.management import call_command from iprestrict import models class Command(BaseCommand): help = 'Replaces the current rules in the DB with the rules in the given fixture file(s).' args = "fixture [fixture ...]" def handle(self, *args, **options): verbosity = int(options.get('verbosity', '1')) self.delete_existing_rules() if verbosity >= 1: self.stdout.write('Successfully deleted rules\n') call_command('loaddata', *args, verbosity=verbosity, interactive=False) def delete_existing_rules(self): models.Rule.objects.all().delete() models.IPRange.objects.all().delete() models.IPGroup.objects.all().delete() PKt[C$-iprestrict/management/commands/reloadrules.pyfrom django.core.management.base import BaseCommand from iprestrict import models class Command(BaseCommand): help = 'Requests the reload of the ip restriction rules from the DB.' def handle(self, *args, **options): verbosity = int(options.get('verbosity', '1')) models.ReloadRulesRequest.request_reload() if verbosity >= 1: self.stdout.write('Successfully requested reload of rules\n') PKt[Ciprestrict/templates/404.htmlPKt[CЄ /iprestrict/templates/iprestrict/test_rules.html{% load staticfiles %} {% load templatetag_handlebars %} IPRestrict - Test Rules

Test Rules

Enter a URL and an IP address below, then click Test.

URL IP Address

Current Rules

URL pattern IP Group Action
Rules will be loaded on first test
{% tplhandlebars "rules-template" %} {{#each rules}} {{#if this.matched }} {{else}} {{/if}} {{this.url_pattern.value}} {{this.ip_group.name}}
({{this.ip_group.ranges}}) {{this.action}} {{/each}} {% endtplhandlebars %} PKt[Ccuu$iprestrict/static/css/test_rules.css body { padding-left: 10px; color: #333333; font-family: ‘Palatino Linotype’, ‘Book Antiqua’, Palatino, serif; font-size: 18px; } h2 { font-size: 1.7em; padding-left: 20px; } p, table { padding-left: 35px; } input, button { font-size: 110%; padding: 5px; } #result { font-size: 1.2em; } #action.Allowed { color: green; } #action.Denied { color: red; } table#rules { width: 75%; border-collapse: collapse; border: 1px solid #333333; } table#rules th { border: 1px solid #333333; padding: 10px; background-color: #333333; color: white; } table#rules td { text-align: center; border: 1px solid #333333; padding: 5px; } tr.ruleMatch { background-color: lightgreen; } td.match { color: green; font-weight: bold; } td.noMatch { color: red; font-weight: bold; } PKt[C1##.iprestrict/static/javascript/lib/handlebars.js// lib/handlebars/base.js var Handlebars = {}; Handlebars.VERSION = "1.0.beta.6"; Handlebars.helpers = {}; Handlebars.partials = {}; Handlebars.registerHelper = function(name, fn, inverse) { if(inverse) { fn.not = inverse; } this.helpers[name] = fn; }; Handlebars.registerPartial = function(name, str) { this.partials[name] = str; }; Handlebars.registerHelper('helperMissing', function(arg) { if(arguments.length === 2) { return undefined; } else { throw new Error("Could not find property '" + arg + "'"); } }); var toString = Object.prototype.toString, functionType = "[object Function]"; Handlebars.registerHelper('blockHelperMissing', function(context, options) { var inverse = options.inverse || function() {}, fn = options.fn; var ret = ""; var type = toString.call(context); if(type === functionType) { context = context.call(this); } if(context === true) { return fn(this); } else if(context === false || context == null) { return inverse(this); } else if(type === "[object Array]") { if(context.length > 0) { for(var i=0, j=context.length; i 0) { for(var i=0, j=context.length; i 2) { expected.push("'" + this.terminals_[p] + "'"); } var errStr = ""; if (this.lexer.showPosition) { errStr = "Parse error on line " + (yylineno + 1) + ":\n" + this.lexer.showPosition() + "\nExpecting " + expected.join(", ") + ", got '" + this.terminals_[symbol] + "'"; } else { errStr = "Parse error on line " + (yylineno + 1) + ": Unexpected " + (symbol == 1?"end of input":"'" + (this.terminals_[symbol] || symbol) + "'"); } this.parseError(errStr, {text: this.lexer.match, token: this.terminals_[symbol] || symbol, line: this.lexer.yylineno, loc: yyloc, expected: expected}); } } if (action[0] instanceof Array && action.length > 1) { throw new Error("Parse Error: multiple actions possible at state: " + state + ", token: " + symbol); } switch (action[0]) { case 1: stack.push(symbol); vstack.push(this.lexer.yytext); lstack.push(this.lexer.yylloc); stack.push(action[1]); symbol = null; if (!preErrorSymbol) { yyleng = this.lexer.yyleng; yytext = this.lexer.yytext; yylineno = this.lexer.yylineno; yyloc = this.lexer.yylloc; if (recovering > 0) recovering--; } else { symbol = preErrorSymbol; preErrorSymbol = null; } break; case 2: len = this.productions_[action[1]][1]; yyval.$ = vstack[vstack.length - len]; yyval._$ = {first_line: lstack[lstack.length - (len || 1)].first_line, last_line: lstack[lstack.length - 1].last_line, first_column: lstack[lstack.length - (len || 1)].first_column, last_column: lstack[lstack.length - 1].last_column}; r = this.performAction.call(yyval, yytext, yyleng, yylineno, this.yy, action[1], vstack, lstack); if (typeof r !== "undefined") { return r; } if (len) { stack = stack.slice(0, -1 * len * 2); vstack = vstack.slice(0, -1 * len); lstack = lstack.slice(0, -1 * len); } stack.push(this.productions_[action[1]][0]); vstack.push(yyval.$); lstack.push(yyval._$); newState = table[stack[stack.length - 2]][stack[stack.length - 1]]; stack.push(newState); break; case 3: return true; } } return true; } };/* Jison generated lexer */ var lexer = (function(){ var lexer = ({EOF:1, parseError:function parseError(str, hash) { if (this.yy.parseError) { this.yy.parseError(str, hash); } else { throw new Error(str); } }, setInput:function (input) { this._input = input; this._more = this._less = this.done = false; this.yylineno = this.yyleng = 0; this.yytext = this.matched = this.match = ''; this.conditionStack = ['INITIAL']; this.yylloc = {first_line:1,first_column:0,last_line:1,last_column:0}; return this; }, input:function () { var ch = this._input[0]; this.yytext+=ch; this.yyleng++; this.match+=ch; this.matched+=ch; var lines = ch.match(/\n/); if (lines) this.yylineno++; this._input = this._input.slice(1); return ch; }, unput:function (ch) { this._input = ch + this._input; return this; }, more:function () { this._more = true; return this; }, pastInput:function () { var past = this.matched.substr(0, this.matched.length - this.match.length); return (past.length > 20 ? '...':'') + past.substr(-20).replace(/\n/g, ""); }, upcomingInput:function () { var next = this.match; if (next.length < 20) { next += this._input.substr(0, 20-next.length); } return (next.substr(0,20)+(next.length > 20 ? '...':'')).replace(/\n/g, ""); }, showPosition:function () { var pre = this.pastInput(); var c = new Array(pre.length + 1).join("-"); return pre + this.upcomingInput() + "\n" + c+"^"; }, next:function () { if (this.done) { return this.EOF; } if (!this._input) this.done = true; var token, match, col, lines; if (!this._more) { this.yytext = ''; this.match = ''; } var rules = this._currentRules(); for (var i=0;i < rules.length; i++) { match = this._input.match(this.rules[rules[i]]); if (match) { lines = match[0].match(/\n.*/g); if (lines) this.yylineno += lines.length; this.yylloc = {first_line: this.yylloc.last_line, last_line: this.yylineno+1, first_column: this.yylloc.last_column, last_column: lines ? lines[lines.length-1].length-1 : this.yylloc.last_column + match[0].length} this.yytext += match[0]; this.match += match[0]; this.matches = match; this.yyleng = this.yytext.length; this._more = false; this._input = this._input.slice(match[0].length); this.matched += match[0]; token = this.performAction.call(this, this.yy, this, rules[i],this.conditionStack[this.conditionStack.length-1]); if (token) return token; else return; } } if (this._input === "") { return this.EOF; } else { this.parseError('Lexical error on line '+(this.yylineno+1)+'. Unrecognized text.\n'+this.showPosition(), {text: "", token: null, line: this.yylineno}); } }, lex:function lex() { var r = this.next(); if (typeof r !== 'undefined') { return r; } else { return this.lex(); } }, begin:function begin(condition) { this.conditionStack.push(condition); }, popState:function popState() { return this.conditionStack.pop(); }, _currentRules:function _currentRules() { return this.conditions[this.conditionStack[this.conditionStack.length-1]].rules; }, topState:function () { return this.conditionStack[this.conditionStack.length-2]; }, pushState:function begin(condition) { this.begin(condition); }}); lexer.performAction = function anonymous(yy,yy_,$avoiding_name_collisions,YY_START) { var YYSTATE=YY_START switch($avoiding_name_collisions) { case 0: if(yy_.yytext.slice(-1) !== "\\") this.begin("mu"); if(yy_.yytext.slice(-1) === "\\") yy_.yytext = yy_.yytext.substr(0,yy_.yyleng-1), this.begin("emu"); if(yy_.yytext) return 14; break; case 1: return 14; break; case 2: this.popState(); return 14; break; case 3: return 24; break; case 4: return 16; break; case 5: return 20; break; case 6: return 19; break; case 7: return 19; break; case 8: return 23; break; case 9: return 23; break; case 10: yy_.yytext = yy_.yytext.substr(3,yy_.yyleng-5); this.popState(); return 15; break; case 11: return 22; break; case 12: return 34; break; case 13: return 33; break; case 14: return 33; break; case 15: return 36; break; case 16: /*ignore whitespace*/ break; case 17: this.popState(); return 18; break; case 18: this.popState(); return 18; break; case 19: yy_.yytext = yy_.yytext.substr(1,yy_.yyleng-2).replace(/\\"/g,'"'); return 28; break; case 20: return 30; break; case 21: return 30; break; case 22: return 29; break; case 23: return 33; break; case 24: yy_.yytext = yy_.yytext.substr(1, yy_.yyleng-2); return 33; break; case 25: return 'INVALID'; break; case 26: return 5; break; } }; lexer.rules = [/^[^\x00]*?(?=(\{\{))/,/^[^\x00]+/,/^[^\x00]{2,}?(?=(\{\{))/,/^\{\{>/,/^\{\{#/,/^\{\{\//,/^\{\{\^/,/^\{\{\s*else\b/,/^\{\{\{/,/^\{\{&/,/^\{\{![\s\S]*?\}\}/,/^\{\{/,/^=/,/^\.(?=[} ])/,/^\.\./,/^[\/.]/,/^\s+/,/^\}\}\}/,/^\}\}/,/^"(\\["]|[^"])*"/,/^true(?=[}\s])/,/^false(?=[}\s])/,/^[0-9]+(?=[}\s])/,/^[a-zA-Z0-9_$-]+(?=[=}\s\/.])/,/^\[[^\]]*\]/,/^./,/^$/]; lexer.conditions = {"mu":{"rules":[3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26],"inclusive":false},"emu":{"rules":[2],"inclusive":false},"INITIAL":{"rules":[0,1,26],"inclusive":true}};return lexer;})() parser.lexer = lexer; return parser; })(); if (typeof require !== 'undefined' && typeof exports !== 'undefined') { exports.parser = handlebars; exports.parse = function () { return handlebars.parse.apply(handlebars, arguments); } exports.main = function commonjsMain(args) { if (!args[1]) throw new Error('Usage: '+args[0]+' FILE'); if (typeof process !== 'undefined') { var source = require('fs').readFileSync(require('path').join(process.cwd(), args[1]), "utf8"); } else { var cwd = require("file").path(require("file").cwd()); var source = cwd.join(args[1]).read({charset: "utf-8"}); } return exports.parser.parse(source); } if (typeof module !== 'undefined' && require.main === module) { exports.main(typeof process !== 'undefined' ? process.argv.slice(1) : require("system").args); } }; ; // lib/handlebars/compiler/base.js Handlebars.Parser = handlebars; Handlebars.parse = function(string) { Handlebars.Parser.yy = Handlebars.AST; return Handlebars.Parser.parse(string); }; Handlebars.print = function(ast) { return new Handlebars.PrintVisitor().accept(ast); }; Handlebars.logger = { DEBUG: 0, INFO: 1, WARN: 2, ERROR: 3, level: 3, // override in the host environment log: function(level, str) {} }; Handlebars.log = function(level, str) { Handlebars.logger.log(level, str); }; ; // lib/handlebars/compiler/ast.js (function() { Handlebars.AST = {}; Handlebars.AST.ProgramNode = function(statements, inverse) { this.type = "program"; this.statements = statements; if(inverse) { this.inverse = new Handlebars.AST.ProgramNode(inverse); } }; Handlebars.AST.MustacheNode = function(params, hash, unescaped) { this.type = "mustache"; this.id = params[0]; this.params = params.slice(1); this.hash = hash; this.escaped = !unescaped; }; Handlebars.AST.PartialNode = function(id, context) { this.type = "partial"; // TODO: disallow complex IDs this.id = id; this.context = context; }; var verifyMatch = function(open, close) { if(open.original !== close.original) { throw new Handlebars.Exception(open.original + " doesn't match " + close.original); } }; Handlebars.AST.BlockNode = function(mustache, program, close) { verifyMatch(mustache.id, close); this.type = "block"; this.mustache = mustache; this.program = program; }; Handlebars.AST.InverseNode = function(mustache, program, close) { verifyMatch(mustache.id, close); this.type = "inverse"; this.mustache = mustache; this.program = program; }; Handlebars.AST.ContentNode = function(string) { this.type = "content"; this.string = string; }; Handlebars.AST.HashNode = function(pairs) { this.type = "hash"; this.pairs = pairs; }; Handlebars.AST.IdNode = function(parts) { this.type = "ID"; this.original = parts.join("."); var dig = [], depth = 0; for(var i=0,l=parts.length; i": ">", '"': """, "'": "'", "`": "`" }; var badChars = /&(?!\w+;)|[<>"'`]/g; var possible = /[&<>"'`]/; var escapeChar = function(chr) { return escape[chr] || "&"; }; Handlebars.Utils = { escapeExpression: function(string) { // don't escape SafeStrings, since they're already safe if (string instanceof Handlebars.SafeString) { return string.toString(); } else if (string == null || string === false) { return ""; } if(!possible.test(string)) { return string; } return string.replace(badChars, escapeChar); }, isEmpty: function(value) { if (typeof value === "undefined") { return true; } else if (value === null) { return true; } else if (value === false) { return true; } else if(Object.prototype.toString.call(value) === "[object Array]" && value.length === 0) { return true; } else { return false; } } }; })();; // lib/handlebars/compiler/compiler.js Handlebars.Compiler = function() {}; Handlebars.JavaScriptCompiler = function() {}; (function(Compiler, JavaScriptCompiler) { Compiler.OPCODE_MAP = { appendContent: 1, getContext: 2, lookupWithHelpers: 3, lookup: 4, append: 5, invokeMustache: 6, appendEscaped: 7, pushString: 8, truthyOrFallback: 9, functionOrFallback: 10, invokeProgram: 11, invokePartial: 12, push: 13, assignToHash: 15, pushStringParam: 16 }; Compiler.MULTI_PARAM_OPCODES = { appendContent: 1, getContext: 1, lookupWithHelpers: 2, lookup: 1, invokeMustache: 3, pushString: 1, truthyOrFallback: 1, functionOrFallback: 1, invokeProgram: 3, invokePartial: 1, push: 1, assignToHash: 1, pushStringParam: 1 }; Compiler.DISASSEMBLE_MAP = {}; for(var prop in Compiler.OPCODE_MAP) { var value = Compiler.OPCODE_MAP[prop]; Compiler.DISASSEMBLE_MAP[value] = prop; } Compiler.multiParamSize = function(code) { return Compiler.MULTI_PARAM_OPCODES[Compiler.DISASSEMBLE_MAP[code]]; }; Compiler.prototype = { compiler: Compiler, disassemble: function() { var opcodes = this.opcodes, opcode, nextCode; var out = [], str, name, value; for(var i=0, l=opcodes.length; i 0) { this.source[1] = this.source[1] + ", " + locals.join(", "); } // Generate minimizer alias mappings if (!this.isChild) { var aliases = [] for (var alias in this.context.aliases) { this.source[1] = this.source[1] + ', ' + alias + '=' + this.context.aliases[alias]; } } if (this.source[1]) { this.source[1] = "var " + this.source[1].substring(2) + ";"; } // Merge children if (!this.isChild) { this.source[1] += '\n' + this.context.programs.join('\n') + '\n'; } if (!this.environment.isSimple) { this.source.push("return buffer;"); } var params = this.isChild ? ["depth0", "data"] : ["Handlebars", "depth0", "helpers", "partials", "data"]; for(var i=0, l=this.environment.depths.list.length; i this.stackVars.length) { this.stackVars.push("stack" + this.stackSlot); } return "stack" + this.stackSlot; }, popStack: function() { return "stack" + this.stackSlot--; }, topStack: function() { return "stack" + this.stackSlot; }, quotedString: function(str) { return '"' + str .replace(/\\/g, '\\\\') .replace(/"/g, '\\"') .replace(/\n/g, '\\n') .replace(/\r/g, '\\r') + '"'; } }; var reservedWords = ( "break else new var" + " case finally return void" + " catch for switch while" + " continue function this with" + " default if throw" + " delete in try" + " do instanceof typeof" + " abstract enum int short" + " boolean export interface static" + " byte extends long super" + " char final native synchronized" + " class float package throws" + " const goto private transient" + " debugger implements protected volatile" + " double import public let yield" ).split(" "); var compilerWords = JavaScriptCompiler.RESERVED_WORDS = {}; for(var i=0, l=reservedWords.length; i=0===c})}function bk(a){var b=bl.split("|"),c=a.createDocumentFragment();if(c.createElement)while(b.length)c.createElement(b.pop());return c}function bC(a,b){return a.getElementsByTagName(b)[0]||a.appendChild(a.ownerDocument.createElement(b))}function bD(a,b){if(b.nodeType!==1||!p.hasData(a))return;var c,d,e,f=p._data(a),g=p._data(b,f),h=f.events;if(h){delete g.handle,g.events={};for(c in h)for(d=0,e=h[c].length;d").appendTo(e.body),c=b.css("display");b.remove();if(c==="none"||c===""){bI=e.body.appendChild(bI||p.extend(e.createElement("iframe"),{frameBorder:0,width:0,height:0}));if(!bJ||!bI.createElement)bJ=(bI.contentWindow||bI.contentDocument).document,bJ.write(""),bJ.close();b=bJ.body.appendChild(bJ.createElement(a)),c=bH(b,"display"),e.body.removeChild(bI)}return bR[a]=c,c}function ch(a,b,c,d){var e;if(p.isArray(b))p.each(b,function(b,e){c||cd.test(a)?d(a,e):ch(a+"["+(typeof e=="object"?b:"")+"]",e,c,d)});else if(!c&&p.type(b)==="object")for(e in b)ch(a+"["+e+"]",b[e],c,d);else d(a,b)}function cy(a){return function(b,c){typeof b!="string"&&(c=b,b="*");var d,e,f,g=b.toLowerCase().split(s),h=0,i=g.length;if(p.isFunction(c))for(;h)[^>]*$|#([\w\-]*)$)/,v=/^<(\w+)\s*\/?>(?:<\/\1>|)$/,w=/^[\],:{}\s]*$/,x=/(?:^|:|,)(?:\s*\[)+/g,y=/\\(?:["\\\/bfnrt]|u[\da-fA-F]{4})/g,z=/"[^"\\\r\n]*"|true|false|null|-?(?:\d\d*\.|)\d+(?:[eE][\-+]?\d+|)/g,A=/^-ms-/,B=/-([\da-z])/gi,C=function(a,b){return(b+"").toUpperCase()},D=function(){e.addEventListener?(e.removeEventListener("DOMContentLoaded",D,!1),p.ready()):e.readyState==="complete"&&(e.detachEvent("onreadystatechange",D),p.ready())},E={};p.fn=p.prototype={constructor:p,init:function(a,c,d){var f,g,h,i;if(!a)return this;if(a.nodeType)return this.context=this[0]=a,this.length=1,this;if(typeof a=="string"){a.charAt(0)==="<"&&a.charAt(a.length-1)===">"&&a.length>=3?f=[null,a,null]:f=u.exec(a);if(f&&(f[1]||!c)){if(f[1])return c=c instanceof p?c[0]:c,i=c&&c.nodeType?c.ownerDocument||c:e,a=p.parseHTML(f[1],i,!0),v.test(f[1])&&p.isPlainObject(c)&&this.attr.call(a,c,!0),p.merge(this,a);g=e.getElementById(f[2]);if(g&&g.parentNode){if(g.id!==f[2])return d.find(a);this.length=1,this[0]=g}return this.context=e,this.selector=a,this}return!c||c.jquery?(c||d).find(a):this.constructor(c).find(a)}return p.isFunction(a)?d.ready(a):(a.selector!==b&&(this.selector=a.selector,this.context=a.context),p.makeArray(a,this))},selector:"",jquery:"1.8.0",length:0,size:function(){return this.length},toArray:function(){return k.call(this)},get:function(a){return a==null?this.toArray():a<0?this[this.length+a]:this[a]},pushStack:function(a,b,c){var d=p.merge(this.constructor(),a);return d.prevObject=this,d.context=this.context,b==="find"?d.selector=this.selector+(this.selector?" ":"")+c:b&&(d.selector=this.selector+"."+b+"("+c+")"),d},each:function(a,b){return p.each(this,a,b)},ready:function(a){return p.ready.promise().done(a),this},eq:function(a){return a=+a,a===-1?this.slice(a):this.slice(a,a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(k.apply(this,arguments),"slice",k.call(arguments).join(","))},map:function(a){return this.pushStack(p.map(this,function(b,c){return a.call(b,c,b)}))},end:function(){return this.prevObject||this.constructor(null)},push:j,sort:[].sort,splice:[].splice},p.fn.init.prototype=p.fn,p.extend=p.fn.extend=function(){var a,c,d,e,f,g,h=arguments[0]||{},i=1,j=arguments.length,k=!1;typeof h=="boolean"&&(k=h,h=arguments[1]||{},i=2),typeof h!="object"&&!p.isFunction(h)&&(h={}),j===i&&(h=this,--i);for(;i0)return;d.resolveWith(e,[p]),p.fn.trigger&&p(e).trigger("ready").off("ready")},isFunction:function(a){return p.type(a)==="function"},isArray:Array.isArray||function(a){return p.type(a)==="array"},isWindow:function(a){return a!=null&&a==a.window},isNumeric:function(a){return!isNaN(parseFloat(a))&&isFinite(a)},type:function(a){return a==null?String(a):E[m.call(a)]||"object"},isPlainObject:function(a){if(!a||p.type(a)!=="object"||a.nodeType||p.isWindow(a))return!1;try{if(a.constructor&&!n.call(a,"constructor")&&!n.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(c){return!1}var d;for(d in a);return d===b||n.call(a,d)},isEmptyObject:function(a){var b;for(b in a)return!1;return!0},error:function(a){throw new Error(a)},parseHTML:function(a,b,c){var d;return!a||typeof a!="string"?null:(typeof b=="boolean"&&(c=b,b=0),b=b||e,(d=v.exec(a))?[b.createElement(d[1])]:(d=p.buildFragment([a],b,c?null:[]),p.merge([],(d.cacheable?p.clone(d.fragment):d.fragment).childNodes)))},parseJSON:function(b){if(!b||typeof b!="string")return null;b=p.trim(b);if(a.JSON&&a.JSON.parse)return a.JSON.parse(b);if(w.test(b.replace(y,"@").replace(z,"]").replace(x,"")))return(new Function("return "+b))();p.error("Invalid JSON: "+b)},parseXML:function(c){var d,e;if(!c||typeof c!="string")return null;try{a.DOMParser?(e=new DOMParser,d=e.parseFromString(c,"text/xml")):(d=new ActiveXObject("Microsoft.XMLDOM"),d.async="false",d.loadXML(c))}catch(f){d=b}return(!d||!d.documentElement||d.getElementsByTagName("parsererror").length)&&p.error("Invalid XML: "+c),d},noop:function(){},globalEval:function(b){b&&r.test(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(A,"ms-").replace(B,C)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toUpperCase()===b.toUpperCase()},each:function(a,c,d){var e,f=0,g=a.length,h=g===b||p.isFunction(a);if(d){if(h){for(e in a)if(c.apply(a[e],d)===!1)break}else for(;f0&&a[0]&&a[i-1]||i===0||p.isArray(a));if(j)for(;h-1)i.splice(c,1),e&&(c<=g&&g--,c<=h&&h--)}),this},has:function(a){return p.inArray(a,i)>-1},empty:function(){return i=[],this},disable:function(){return i=j=c=b,this},disabled:function(){return!i},lock:function(){return j=b,c||l.disable(),this},locked:function(){return!j},fireWith:function(a,b){return b=b||[],b=[a,b.slice?b.slice():b],i&&(!d||j)&&(e?j.push(b):k(b)),this},fire:function(){return l.fireWith(this,arguments),this},fired:function(){return!!d}};return l},p.extend({Deferred:function(a){var b=[["resolve","done",p.Callbacks("once memory"),"resolved"],["reject","fail",p.Callbacks("once memory"),"rejected"],["notify","progress",p.Callbacks("memory")]],c="pending",d={state:function(){return c},always:function(){return e.done(arguments).fail(arguments),this},then:function(){var a=arguments;return p.Deferred(function(c){p.each(b,function(b,d){var f=d[0],g=a[b];e[d[1]](p.isFunction(g)?function(){var a=g.apply(this,arguments);a&&p.isFunction(a.promise)?a.promise().done(c.resolve).fail(c.reject).progress(c.notify):c[f+"With"](this===e?c:this,[a])}:c[f])}),a=null}).promise()},promise:function(a){return typeof a=="object"?p.extend(a,d):d}},e={};return d.pipe=d.then,p.each(b,function(a,f){var g=f[2],h=f[3];d[f[1]]=g.add,h&&g.add(function(){c=h},b[a^1][2].disable,b[2][2].lock),e[f[0]]=g.fire,e[f[0]+"With"]=g.fireWith}),d.promise(e),a&&a.call(e,e),e},when:function(a){var b=0,c=k.call(arguments),d=c.length,e=d!==1||a&&p.isFunction(a.promise)?d:0,f=e===1?a:p.Deferred(),g=function(a,b,c){return function(d){b[a]=this,c[a]=arguments.length>1?k.call(arguments):d,c===h?f.notifyWith(b,c):--e||f.resolveWith(b,c)}},h,i,j;if(d>1){h=new Array(d),i=new Array(d),j=new Array(d);for(;b
a",c=n.getElementsByTagName("*"),d=n.getElementsByTagName("a")[0],d.style.cssText="top:1px;float:left;opacity:.5";if(!c||!c.length||!d)return{};f=e.createElement("select"),g=f.appendChild(e.createElement("option")),h=n.getElementsByTagName("input")[0],b={leadingWhitespace:n.firstChild.nodeType===3,tbody:!n.getElementsByTagName("tbody").length,htmlSerialize:!!n.getElementsByTagName("link").length,style:/top/.test(d.getAttribute("style")),hrefNormalized:d.getAttribute("href")==="/a",opacity:/^0.5/.test(d.style.opacity),cssFloat:!!d.style.cssFloat,checkOn:h.value==="on",optSelected:g.selected,getSetAttribute:n.className!=="t",enctype:!!e.createElement("form").enctype,html5Clone:e.createElement("nav").cloneNode(!0).outerHTML!=="<:nav>",boxModel:e.compatMode==="CSS1Compat",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0,boxSizingReliable:!0,pixelPosition:!1},h.checked=!0,b.noCloneChecked=h.cloneNode(!0).checked,f.disabled=!0,b.optDisabled=!g.disabled;try{delete n.test}catch(o){b.deleteExpando=!1}!n.addEventListener&&n.attachEvent&&n.fireEvent&&(n.attachEvent("onclick",m=function(){b.noCloneEvent=!1}),n.cloneNode(!0).fireEvent("onclick"),n.detachEvent("onclick",m)),h=e.createElement("input"),h.value="t",h.setAttribute("type","radio"),b.radioValue=h.value==="t",h.setAttribute("checked","checked"),h.setAttribute("name","t"),n.appendChild(h),i=e.createDocumentFragment(),i.appendChild(n.lastChild),b.checkClone=i.cloneNode(!0).cloneNode(!0).lastChild.checked,b.appendChecked=h.checked,i.removeChild(h),i.appendChild(n);if(n.attachEvent)for(k in{submit:!0,change:!0,focusin:!0})j="on"+k,l=j in n,l||(n.setAttribute(j,"return;"),l=typeof n[j]=="function"),b[k+"Bubbles"]=l;return p(function(){var c,d,f,g,h="padding:0;margin:0;border:0;display:block;overflow:hidden;",i=e.getElementsByTagName("body")[0];if(!i)return;c=e.createElement("div"),c.style.cssText="visibility:hidden;border:0;width:0;height:0;position:static;top:0;margin-top:1px",i.insertBefore(c,i.firstChild),d=e.createElement("div"),c.appendChild(d),d.innerHTML="
t
",f=d.getElementsByTagName("td"),f[0].style.cssText="padding:0;margin:0;border:0;display:none",l=f[0].offsetHeight===0,f[0].style.display="",f[1].style.display="none",b.reliableHiddenOffsets=l&&f[0].offsetHeight===0,d.innerHTML="",d.style.cssText="box-sizing:border-box;-moz-box-sizing:border-box;-webkit-box-sizing:border-box;padding:1px;border:1px;display:block;width:4px;margin-top:1%;position:absolute;top:1%;",b.boxSizing=d.offsetWidth===4,b.doesNotIncludeMarginInBodyOffset=i.offsetTop!==1,a.getComputedStyle&&(b.pixelPosition=(a.getComputedStyle(d,null)||{}).top!=="1%",b.boxSizingReliable=(a.getComputedStyle(d,null)||{width:"4px"}).width==="4px",g=e.createElement("div"),g.style.cssText=d.style.cssText=h,g.style.marginRight=g.style.width="0",d.style.width="1px",d.appendChild(g),b.reliableMarginRight=!parseFloat((a.getComputedStyle(g,null)||{}).marginRight)),typeof d.style.zoom!="undefined"&&(d.innerHTML="",d.style.cssText=h+"width:1px;padding:1px;display:inline;zoom:1",b.inlineBlockNeedsLayout=d.offsetWidth===3,d.style.display="block",d.style.overflow="visible",d.innerHTML="
",d.firstChild.style.width="5px",b.shrinkWrapBlocks=d.offsetWidth!==3,c.style.zoom=1),i.removeChild(c),c=d=f=g=null}),i.removeChild(n),c=d=f=g=h=i=n=null,b}();var H=/^(?:\{.*\}|\[.*\])$/,I=/([A-Z])/g;p.extend({cache:{},deletedIds:[],uuid:0,expando:"jQuery"+(p.fn.jquery+Math.random()).replace(/\D/g,""),noData:{embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",applet:!0},hasData:function(a){return a=a.nodeType?p.cache[a[p.expando]]:a[p.expando],!!a&&!K(a)},data:function(a,c,d,e){if(!p.acceptData(a))return;var f,g,h=p.expando,i=typeof c=="string",j=a.nodeType,k=j?p.cache:a,l=j?a[h]:a[h]&&h;if((!l||!k[l]||!e&&!k[l].data)&&i&&d===b)return;l||(j?a[h]=l=p.deletedIds.pop()||++p.uuid:l=h),k[l]||(k[l]={},j||(k[l].toJSON=p.noop));if(typeof c=="object"||typeof c=="function")e?k[l]=p.extend(k[l],c):k[l].data=p.extend(k[l].data,c);return f=k[l],e||(f.data||(f.data={}),f=f.data),d!==b&&(f[p.camelCase(c)]=d),i?(g=f[c],g==null&&(g=f[p.camelCase(c)])):g=f,g},removeData:function(a,b,c){if(!p.acceptData(a))return;var d,e,f,g=a.nodeType,h=g?p.cache:a,i=g?a[p.expando]:p.expando;if(!h[i])return;if(b){d=c?h[i]:h[i].data;if(d){p.isArray(b)||(b in d?b=[b]:(b=p.camelCase(b),b in d?b=[b]:b=b.split(" ")));for(e=0,f=b.length;e1,null,!1))},removeData:function(a){return this.each(function(){p.removeData(this,a)})}}),p.extend({queue:function(a,b,c){var d;if(a)return b=(b||"fx")+"queue",d=p._data(a,b),c&&(!d||p.isArray(c)?d=p._data(a,b,p.makeArray(c)):d.push(c)),d||[]},dequeue:function(a,b){b=b||"fx";var c=p.queue(a,b),d=c.shift(),e=p._queueHooks(a,b),f=function(){p.dequeue(a,b)};d==="inprogress"&&(d=c.shift()),d&&(b==="fx"&&c.unshift("inprogress"),delete e.stop,d.call(a,f,e)),!c.length&&e&&e.empty.fire()},_queueHooks:function(a,b){var c=b+"queueHooks";return p._data(a,c)||p._data(a,c,{empty:p.Callbacks("once memory").add(function(){p.removeData(a,b+"queue",!0),p.removeData(a,c,!0)})})}}),p.fn.extend({queue:function(a,c){var d=2;return typeof a!="string"&&(c=a,a="fx",d--),arguments.length1)},removeAttr:function(a){return this.each(function(){p.removeAttr(this,a)})},prop:function(a,b){return p.access(this,p.prop,a,b,arguments.length>1)},removeProp:function(a){return a=p.propFix[a]||a,this.each(function(){try{this[a]=b,delete this[a]}catch(c){}})},addClass:function(a){var b,c,d,e,f,g,h;if(p.isFunction(a))return this.each(function(b){p(this).addClass(a.call(this,b,this.className))});if(a&&typeof a=="string"){b=a.split(s);for(c=0,d=this.length;c-1)d=d.replace(" "+c[f]+" "," ");e.className=a?p.trim(d):""}}}return this},toggleClass:function(a,b){var c=typeof a,d=typeof b=="boolean";return p.isFunction(a)?this.each(function(c){p(this).toggleClass(a.call(this,c,this.className,b),b)}):this.each(function(){if(c==="string"){var e,f=0,g=p(this),h=b,i=a.split(s);while(e=i[f++])h=d?h:!g.hasClass(e),g[h?"addClass":"removeClass"](e)}else if(c==="undefined"||c==="boolean")this.className&&p._data(this,"__className__",this.className),this.className=this.className||a===!1?"":p._data(this,"__className__")||""})},hasClass:function(a){var b=" "+a+" ",c=0,d=this.length;for(;c-1)return!0;return!1},val:function(a){var c,d,e,f=this[0];if(!arguments.length){if(f)return c=p.valHooks[f.type]||p.valHooks[f.nodeName.toLowerCase()],c&&"get"in c&&(d=c.get(f,"value"))!==b?d:(d=f.value,typeof d=="string"?d.replace(P,""):d==null?"":d);return}return e=p.isFunction(a),this.each(function(d){var f,g=p(this);if(this.nodeType!==1)return;e?f=a.call(this,d,g.val()):f=a,f==null?f="":typeof f=="number"?f+="":p.isArray(f)&&(f=p.map(f,function(a){return a==null?"":a+""})),c=p.valHooks[this.type]||p.valHooks[this.nodeName.toLowerCase()];if(!c||!("set"in c)||c.set(this,f,"value")===b)this.value=f})}}),p.extend({valHooks:{option:{get:function(a){var b=a.attributes.value;return!b||b.specified?a.value:a.text}},select:{get:function(a){var b,c,d,e,f=a.selectedIndex,g=[],h=a.options,i=a.type==="select-one";if(f<0)return null;c=i?f:0,d=i?f+1:h.length;for(;c=0}),c.length||(a.selectedIndex=-1),c}}},attrFn:{},attr:function(a,c,d,e){var f,g,h,i=a.nodeType;if(!a||i===3||i===8||i===2)return;if(e&&p.isFunction(p.fn[c]))return p(a)[c](d);if(typeof a.getAttribute=="undefined")return p.prop(a,c,d);h=i!==1||!p.isXMLDoc(a),h&&(c=c.toLowerCase(),g=p.attrHooks[c]||(T.test(c)?M:L));if(d!==b){if(d===null){p.removeAttr(a,c);return}return g&&"set"in g&&h&&(f=g.set(a,d,c))!==b?f:(a.setAttribute(c,""+d),d)}return g&&"get"in g&&h&&(f=g.get(a,c))!==null?f:(f=a.getAttribute(c),f===null?b:f)},removeAttr:function(a,b){var c,d,e,f,g=0;if(b&&a.nodeType===1){d=b.split(s);for(;g=0}})});var V=/^(?:textarea|input|select)$/i,W=/^([^\.]*|)(?:\.(.+)|)$/,X=/(?:^|\s)hover(\.\S+|)\b/,Y=/^key/,Z=/^(?:mouse|contextmenu)|click/,$=/^(?:focusinfocus|focusoutblur)$/,_=function(a){return p.event.special.hover?a:a.replace(X,"mouseenter$1 mouseleave$1")};p.event={add:function(a,c,d,e,f){var g,h,i,j,k,l,m,n,o,q,r;if(a.nodeType===3||a.nodeType===8||!c||!d||!(g=p._data(a)))return;d.handler&&(o=d,d=o.handler,f=o.selector),d.guid||(d.guid=p.guid++),i=g.events,i||(g.events=i={}),h=g.handle,h||(g.handle=h=function(a){return typeof p!="undefined"&&(!a||p.event.triggered!==a.type)?p.event.dispatch.apply(h.elem,arguments):b},h.elem=a),c=p.trim(_(c)).split(" ");for(j=0;j=0&&(s=s.slice(0,-1),i=!0),s.indexOf(".")>=0&&(t=s.split("."),s=t.shift(),t.sort());if((!f||p.event.customEvent[s])&&!p.event.global[s])return;c=typeof c=="object"?c[p.expando]?c:new p.Event(s,c):new p.Event(s),c.type=s,c.isTrigger=!0,c.exclusive=i,c.namespace=t.join("."),c.namespace_re=c.namespace?new RegExp("(^|\\.)"+t.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,m=s.indexOf(":")<0?"on"+s:"";if(!f){h=p.cache;for(j in h)h[j].events&&h[j].events[s]&&p.event.trigger(c,d,h[j].handle.elem,!0);return}c.result=b,c.target||(c.target=f),d=d!=null?p.makeArray(d):[],d.unshift(c),n=p.event.special[s]||{};if(n.trigger&&n.trigger.apply(f,d)===!1)return;q=[[f,n.bindType||s]];if(!g&&!n.noBubble&&!p.isWindow(f)){r=n.delegateType||s,k=$.test(r+s)?f:f.parentNode;for(l=f;k;k=k.parentNode)q.push([k,r]),l=k;l===(f.ownerDocument||e)&&q.push([l.defaultView||l.parentWindow||a,r])}for(j=0;jq&&u.push({elem:this,matches:o.slice(q)});for(d=0;d0?this.on(b,null,a,c):this.trigger(b)},Y.test(b)&&(p.event.fixHooks[b]=p.event.keyHooks),Z.test(b)&&(p.event.fixHooks[b]=p.event.mouseHooks)}),function(a,b){function bd(a,b,c,d){var e=0,f=b.length;for(;e0?h(g,c,f):[]}function bf(a,c,d,e,f){var g,h,i,j,k,l,m,n,p=0,q=f.length,s=L.POS,t=new RegExp("^"+s.source+"(?!"+r+")","i"),u=function(){var a=1,c=arguments.length-2;for(;ai){m=a.slice(i,g.index),i=n,l=[c],B.test(m)&&(k&&(l=k),k=e);if(h=H.test(m))m=m.slice(0,-5).replace(B,"$&*");g.length>1&&g[0].replace(t,u),k=be(m,g[1],g[2],l,k,h)}}k?(j=j.concat(k),(m=a.slice(i))&&m!==")"?B.test(m)?bd(m,j,d,e):Z(m,c,d,e?e.concat(k):k):o.apply(d,j)):Z(a,c,d,e)}return q===1?d:Z.uniqueSort(d)}function bg(a,b,c){var d,e,f,g=[],i=0,j=D.exec(a),k=!j.pop()&&!j.pop(),l=k&&a.match(C)||[""],m=$.preFilter,n=$.filter,o=!c&&b!==h;for(;(e=l[i])!=null&&k;i++){g.push(d=[]),o&&(e=" "+e);while(e){k=!1;if(j=B.exec(e))e=e.slice(j[0].length),k=d.push({part:j.pop().replace(A," "),captures:j});for(f in n)(j=L[f].exec(e))&&(!m[f]||(j=m[f](j,b,c)))&&(e=e.slice(j.shift().length),k=d.push({part:f,captures:j}));if(!k)break}}return k||Z.error(a),g}function bh(a,b,e){var f=b.dir,g=m++;return a||(a=function(a){return a===e}),b.first?function(b,c){while(b=b[f])if(b.nodeType===1)return a(b,c)&&b}:function(b,e){var h,i=g+"."+d,j=i+"."+c;while(b=b[f])if(b.nodeType===1){if((h=b[q])===j)return b.sizset;if(typeof h=="string"&&h.indexOf(i)===0){if(b.sizset)return b}else{b[q]=j;if(a(b,e))return b.sizset=!0,b;b.sizset=!1}}}}function bi(a,b){return a?function(c,d){var e=b(c,d);return e&&a(e===!0?c:e,d)}:b}function bj(a,b,c){var d,e,f=0;for(;d=a[f];f++)$.relative[d.part]?e=bh(e,$.relative[d.part],b):(d.captures.push(b,c),e=bi(e,$.filter[d.part].apply(null,d.captures)));return e}function bk(a){return function(b,c){var d,e=0;for(;d=a[e];e++)if(d(b,c))return!0;return!1}}var c,d,e,f,g,h=a.document,i=h.documentElement,j="undefined",k=!1,l=!0,m=0,n=[].slice,o=[].push,q=("sizcache"+Math.random()).replace(".",""),r="[\\x20\\t\\r\\n\\f]",s="(?:\\\\.|[-\\w]|[^\\x00-\\xa0])+",t=s.replace("w","w#"),u="([*^$|!~]?=)",v="\\["+r+"*("+s+")"+r+"*(?:"+u+r+"*(?:(['\"])((?:\\\\.|[^\\\\])*?)\\3|("+t+")|)|)"+r+"*\\]",w=":("+s+")(?:\\((?:(['\"])((?:\\\\.|[^\\\\])*?)\\2|((?:[^,]|\\\\,|(?:,(?=[^\\[]*\\]))|(?:,(?=[^\\(]*\\))))*))\\)|)",x=":(nth|eq|gt|lt|first|last|even|odd)(?:\\((\\d*)\\)|)(?=[^-]|$)",y=r+"*([\\x20\\t\\r\\n\\f>+~])"+r+"*",z="(?=[^\\x20\\t\\r\\n\\f])(?:\\\\.|"+v+"|"+w.replace(2,7)+"|[^\\\\(),])+",A=new RegExp("^"+r+"+|((?:^|[^\\\\])(?:\\\\.)*)"+r+"+$","g"),B=new RegExp("^"+y),C=new RegExp(z+"?(?="+r+"*,|$)","g"),D=new RegExp("^(?:(?!,)(?:(?:^|,)"+r+"*"+z+")*?|"+r+"*(.*?))(\\)|$)"),E=new RegExp(z.slice(19,-6)+"\\x20\\t\\r\\n\\f>+~])+|"+y,"g"),F=/^(?:#([\w\-]+)|(\w+)|\.([\w\-]+))$/,G=/[\x20\t\r\n\f]*[+~]/,H=/:not\($/,I=/h\d/i,J=/input|select|textarea|button/i,K=/\\(?!\\)/g,L={ID:new RegExp("^#("+s+")"),CLASS:new RegExp("^\\.("+s+")"),NAME:new RegExp("^\\[name=['\"]?("+s+")['\"]?\\]"),TAG:new RegExp("^("+s.replace("[-","[-\\*")+")"),ATTR:new RegExp("^"+v),PSEUDO:new RegExp("^"+w),CHILD:new RegExp("^:(only|nth|last|first)-child(?:\\("+r+"*(even|odd|(([+-]|)(\\d*)n|)"+r+"*(?:([+-]|)"+r+"*(\\d+)|))"+r+"*\\)|)","i"),POS:new RegExp(x,"ig"),needsContext:new RegExp("^"+r+"*[>+~]|"+x,"i")},M={},N=[],O={},P=[],Q=function(a){return a.sizzleFilter=!0,a},R=function(a){return function(b){return b.nodeName.toLowerCase()==="input"&&b.type===a}},S=function(a){return function(b){var c=b.nodeName.toLowerCase();return(c==="input"||c==="button")&&b.type===a}},T=function(a){var b=!1,c=h.createElement("div");try{b=a(c)}catch(d){}return c=null,b},U=T(function(a){a.innerHTML="";var b=typeof a.lastChild.getAttribute("multiple");return b!=="boolean"&&b!=="string"}),V=T(function(a){a.id=q+0,a.innerHTML="
",i.insertBefore(a,i.firstChild);var b=h.getElementsByName&&h.getElementsByName(q).length===2+h.getElementsByName(q+0).length;return g=!h.getElementById(q),i.removeChild(a),b}),W=T(function(a){return a.appendChild(h.createComment("")),a.getElementsByTagName("*").length===0}),X=T(function(a){return a.innerHTML="",a.firstChild&&typeof a.firstChild.getAttribute!==j&&a.firstChild.getAttribute("href")==="#"}),Y=T(function(a){return a.innerHTML="",!a.getElementsByClassName||a.getElementsByClassName("e").length===0?!1:(a.lastChild.className="e",a.getElementsByClassName("e").length!==1)}),Z=function(a,b,c,d){c=c||[],b=b||h;var e,f,g,i,j=b.nodeType;if(j!==1&&j!==9)return[];if(!a||typeof a!="string")return c;g=ba(b);if(!g&&!d)if(e=F.exec(a))if(i=e[1]){if(j===9){f=b.getElementById(i);if(!f||!f.parentNode)return c;if(f.id===i)return c.push(f),c}else if(b.ownerDocument&&(f=b.ownerDocument.getElementById(i))&&bb(b,f)&&f.id===i)return c.push(f),c}else{if(e[2])return o.apply(c,n.call(b.getElementsByTagName(a),0)),c;if((i=e[3])&&Y&&b.getElementsByClassName)return o.apply(c,n.call(b.getElementsByClassName(i),0)),c}return bm(a,b,c,d,g)},$=Z.selectors={cacheLength:50,match:L,order:["ID","TAG"],attrHandle:{},createPseudo:Q,find:{ID:g?function(a,b,c){if(typeof b.getElementById!==j&&!c){var d=b.getElementById(a);return d&&d.parentNode?[d]:[]}}:function(a,c,d){if(typeof c.getElementById!==j&&!d){var e=c.getElementById(a);return e?e.id===a||typeof e.getAttributeNode!==j&&e.getAttributeNode("id").value===a?[e]:b:[]}},TAG:W?function(a,b){if(typeof b.getElementsByTagName!==j)return b.getElementsByTagName(a)}:function(a,b){var c=b.getElementsByTagName(a);if(a==="*"){var d,e=[],f=0;for(;d=c[f];f++)d.nodeType===1&&e.push(d);return e}return c}},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(a){return a[1]=a[1].replace(K,""),a[3]=(a[4]||a[5]||"").replace(K,""),a[2]==="~="&&(a[3]=" "+a[3]+" "),a.slice(0,4)},CHILD:function(a){return a[1]=a[1].toLowerCase(),a[1]==="nth"?(a[2]||Z.error(a[0]),a[3]=+(a[3]?a[4]+(a[5]||1):2*(a[2]==="even"||a[2]==="odd")),a[4]=+(a[6]+a[7]||a[2]==="odd")):a[2]&&Z.error(a[0]),a},PSEUDO:function(a){var b,c=a[4];return L.CHILD.test(a[0])?null:(c&&(b=D.exec(c))&&b.pop()&&(a[0]=a[0].slice(0,b[0].length-c.length-1),c=b[0].slice(0,-1)),a.splice(2,3,c||a[3]),a)}},filter:{ID:g?function(a){return a=a.replace(K,""),function(b){return b.getAttribute("id")===a}}:function(a){return a=a.replace(K,""),function(b){var c=typeof b.getAttributeNode!==j&&b.getAttributeNode("id");return c&&c.value===a}},TAG:function(a){return a==="*"?function(){return!0}:(a=a.replace(K,"").toLowerCase(),function(b){return b.nodeName&&b.nodeName.toLowerCase()===a})},CLASS:function(a){var b=M[a];return b||(b=M[a]=new RegExp("(^|"+r+")"+a+"("+r+"|$)"),N.push(a),N.length>$.cacheLength&&delete M[N.shift()]),function(a){return b.test(a.className||typeof a.getAttribute!==j&&a.getAttribute("class")||"")}},ATTR:function(a,b,c){return b?function(d){var e=Z.attr(d,a),f=e+"";if(e==null)return b==="!=";switch(b){case"=":return f===c;case"!=":return f!==c;case"^=":return c&&f.indexOf(c)===0;case"*=":return c&&f.indexOf(c)>-1;case"$=":return c&&f.substr(f.length-c.length)===c;case"~=":return(" "+f+" ").indexOf(c)>-1;case"|=":return f===c||f.substr(0,c.length+1)===c+"-"}}:function(b){return Z.attr(b,a)!=null}},CHILD:function(a,b,c,d){if(a==="nth"){var e=m++;return function(a){var b,f,g=0,h=a;if(c===1&&d===0)return!0;b=a.parentNode;if(b&&(b[q]!==e||!a.sizset)){for(h=b.firstChild;h;h=h.nextSibling)if(h.nodeType===1){h.sizset=++g;if(h===a)break}b[q]=e}return f=a.sizset-d,c===0?f===0:f%c===0&&f/c>=0}}return function(b){var c=b;switch(a){case"only":case"first":while(c=c.previousSibling)if(c.nodeType===1)return!1;if(a==="first")return!0;c=b;case"last":while(c=c.nextSibling)if(c.nodeType===1)return!1;return!0}}},PSEUDO:function(a,b,c,d){var e=$.pseudos[a]||$.pseudos[a.toLowerCase()];return e||Z.error("unsupported pseudo: "+a),e.sizzleFilter?e(b,c,d):e}},pseudos:{not:Q(function(a,b,c){var d=bl(a.replace(A,"$1"),b,c);return function(a){return!d(a)}}),enabled:function(a){return a.disabled===!1},disabled:function(a){return a.disabled===!0},checked:function(a){var b=a.nodeName.toLowerCase();return b==="input"&&!!a.checked||b==="option"&&!!a.selected},selected:function(a){return a.parentNode&&a.parentNode.selectedIndex,a.selected===!0},parent:function(a){return!$.pseudos.empty(a)},empty:function(a){var b;a=a.firstChild;while(a){if(a.nodeName>"@"||(b=a.nodeType)===3||b===4)return!1;a=a.nextSibling}return!0},contains:Q(function(a){return function(b){return(b.textContent||b.innerText||bc(b)).indexOf(a)>-1}}),has:Q(function(a){return function(b){return Z(a,b).length>0}}),header:function(a){return I.test(a.nodeName)},text:function(a){var b,c;return a.nodeName.toLowerCase()==="input"&&(b=a.type)==="text"&&((c=a.getAttribute("type"))==null||c.toLowerCase()===b)},radio:R("radio"),checkbox:R("checkbox"),file:R("file"),password:R("password"),image:R("image"),submit:S("submit"),reset:S("reset"),button:function(a){var b=a.nodeName.toLowerCase();return b==="input"&&a.type==="button"||b==="button"},input:function(a){return J.test(a.nodeName)},focus:function(a){var b=a.ownerDocument;return a===b.activeElement&&(!b.hasFocus||b.hasFocus())&&(!!a.type||!!a.href)},active:function(a){return a===a.ownerDocument.activeElement}},setFilters:{first:function(a,b,c){return c?a.slice(1):[a[0]]},last:function(a,b,c){var d=a.pop();return c?a:[d]},even:function(a,b,c){var d=[],e=c?1:0,f=a.length;for(;e$.cacheLength&&delete O[P.shift()],g};Z.matches=function(a,b){return Z(a,null,null,b)},Z.matchesSelector=function(a,b){return Z(b,null,null,[a]).length>0};var bm=function(a,b,e,f,g){a=a.replace(A,"$1");var h,i,j,k,l,m,p,q,r,s=a.match(C),t=a.match(E),u=b.nodeType;if(L.POS.test(a))return bf(a,b,e,f,s);if(f)h=n.call(f,0);else if(s&&s.length===1){if(t.length>1&&u===9&&!g&&(s=L.ID.exec(t[0]))){b=$.find.ID(s[1],b,g)[0];if(!b)return e;a=a.slice(t.shift().length)}q=(s=G.exec(t[0]))&&!s.index&&b.parentNode||b,r=t.pop(),m=r.split(":not")[0];for(j=0,k=$.order.length;j",a.querySelectorAll("[selected]").length||e.push("\\["+r+"*(?:checked|disabled|ismap|multiple|readonly|selected|value)"),a.querySelectorAll(":checked").length||e.push(":checked")}),T(function(a){a.innerHTML="

",a.querySelectorAll("[test^='']").length&&e.push("[*^$]="+r+"*(?:\"\"|'')"),a.innerHTML="",a.querySelectorAll(":enabled").length||e.push(":enabled",":disabled")}),e=e.length&&new RegExp(e.join("|")),bm=function(a,d,f,g,h){if(!g&&!h&&(!e||!e.test(a)))if(d.nodeType===9)try{return o.apply(f,n.call(d.querySelectorAll(a),0)),f}catch(i){}else if(d.nodeType===1&&d.nodeName.toLowerCase()!=="object"){var j=d.getAttribute("id"),k=j||q,l=G.test(a)&&d.parentNode||d;j?k=k.replace(c,"\\$&"):d.setAttribute("id",k);try{return o.apply(f,n.call(l.querySelectorAll(a.replace(C,"[id='"+k+"'] $&")),0)),f}catch(i){}finally{j||d.removeAttribute("id")}}return b(a,d,f,g,h)},g&&(T(function(b){a=g.call(b,"div");try{g.call(b,"[test!='']:sizzle"),f.push($.match.PSEUDO)}catch(c){}}),f=new RegExp(f.join("|")),Z.matchesSelector=function(b,c){c=c.replace(d,"='$1']");if(!ba(b)&&!f.test(c)&&(!e||!e.test(c)))try{var h=g.call(b,c);if(h||a||b.document&&b.document.nodeType!==11)return h}catch(i){}return Z(c,null,null,[b]).length>0})}(),Z.attr=p.attr,p.find=Z,p.expr=Z.selectors,p.expr[":"]=p.expr.pseudos,p.unique=Z.uniqueSort,p.text=Z.getText,p.isXMLDoc=Z.isXML,p.contains=Z.contains}(a);var bc=/Until$/,bd=/^(?:parents|prev(?:Until|All))/,be=/^.[^:#\[\.,]*$/,bf=p.expr.match.needsContext,bg={children:!0,contents:!0,next:!0,prev:!0};p.fn.extend({find:function(a){var b,c,d,e,f,g,h=this;if(typeof a!="string")return p(a).filter(function(){for(b=0,c=h.length;b0)for(e=d;e=0:p.filter(a,this).length>0:this.filter(a).length>0)},closest:function(a,b){var c,d=0,e=this.length,f=[],g=bf.test(a)||typeof a!="string"?p(a,b||this.context):0;for(;d-1:p.find.matchesSelector(c,a)){f.push(c);break}c=c.parentNode}}return f=f.length>1?p.unique(f):f,this.pushStack(f,"closest",a)},index:function(a){return a?typeof a=="string"?p.inArray(this[0],p(a)):p.inArray(a.jquery?a[0]:a,this):this[0]&&this[0].parentNode?this.prevAll().length:-1},add:function(a,b){var c=typeof a=="string"?p(a,b):p.makeArray(a&&a.nodeType?[a]:a),d=p.merge(this.get(),c);return this.pushStack(bh(c[0])||bh(d[0])?d:p.unique(d))},addBack:function(a){return this.add(a==null?this.prevObject:this.prevObject.filter(a))}}),p.fn.andSelf=p.fn.addBack,p.each({parent:function(a){var b=a.parentNode;return b&&b.nodeType!==11?b:null},parents:function(a){return p.dir(a,"parentNode")},parentsUntil:function(a,b,c){return p.dir(a,"parentNode",c)},next:function(a){return bi(a,"nextSibling")},prev:function(a){return bi(a,"previousSibling")},nextAll:function(a){return p.dir(a,"nextSibling")},prevAll:function(a){return p.dir(a,"previousSibling")},nextUntil:function(a,b,c){return p.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return p.dir(a,"previousSibling",c)},siblings:function(a){return p.sibling((a.parentNode||{}).firstChild,a)},children:function(a){return p.sibling(a.firstChild)},contents:function(a){return p.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:p.merge([],a.childNodes)}},function(a,b){p.fn[a]=function(c,d){var e=p.map(this,b,c);return bc.test(a)||(d=c),d&&typeof d=="string"&&(e=p.filter(d,e)),e=this.length>1&&!bg[a]?p.unique(e):e,this.length>1&&bd.test(a)&&(e=e.reverse()),this.pushStack(e,a,k.call(arguments).join(","))}}),p.extend({filter:function(a,b,c){return c&&(a=":not("+a+")"),b.length===1?p.find.matchesSelector(b[0],a)?[b[0]]:[]:p.find.matches(a,b)},dir:function(a,c,d){var e=[],f=a[c];while(f&&f.nodeType!==9&&(d===b||f.nodeType!==1||!p(f).is(d)))f.nodeType===1&&e.push(f),f=f[c];return e},sibling:function(a,b){var c=[];for(;a;a=a.nextSibling)a.nodeType===1&&a!==b&&c.push(a);return c}});var bl="abbr|article|aside|audio|bdi|canvas|data|datalist|details|figcaption|figure|footer|header|hgroup|mark|meter|nav|output|progress|section|summary|time|video",bm=/ jQuery\d+="(?:null|\d+)"/g,bn=/^\s+/,bo=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/gi,bp=/<([\w:]+)/,bq=/]","i"),bv=/^(?:checkbox|radio)$/,bw=/checked\s*(?:[^=]|=\s*.checked.)/i,bx=/\/(java|ecma)script/i,by=/^\s*\s*$/g,bz={option:[1,""],legend:[1,"
","
"],thead:[1,"","
"],tr:[2,"","
"],td:[3,"","
"],col:[2,"","
"],area:[1,"",""],_default:[0,"",""]},bA=bk(e),bB=bA.appendChild(e.createElement("div"));bz.optgroup=bz.option,bz.tbody=bz.tfoot=bz.colgroup=bz.caption=bz.thead,bz.th=bz.td,p.support.htmlSerialize||(bz._default=[1,"X
","
"]),p.fn.extend({text:function(a){return p.access(this,function(a){return a===b?p.text(this):this.empty().append((this[0]&&this[0].ownerDocument||e).createTextNode(a))},null,a,arguments.length)},wrapAll:function(a){if(p.isFunction(a))return this.each(function(b){p(this).wrapAll(a.call(this,b))});if(this[0]){var b=p(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&a.firstChild.nodeType===1)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){return p.isFunction(a)?this.each(function(b){p(this).wrapInner(a.call(this,b))}):this.each(function(){var b=p(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){var b=p.isFunction(a);return this.each(function(c){p(this).wrapAll(b?a.call(this,c):a)})},unwrap:function(){return this.parent().each(function(){p.nodeName(this,"body")||p(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,!0,function(a){(this.nodeType===1||this.nodeType===11)&&this.appendChild(a)})},prepend:function(){return this.domManip(arguments,!0,function(a){(this.nodeType===1||this.nodeType===11)&&this.insertBefore(a,this.firstChild)})},before:function(){if(!bh(this[0]))return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this)});if(arguments.length){var a=p.clean(arguments);return this.pushStack(p.merge(a,this),"before",this.selector)}},after:function(){if(!bh(this[0]))return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this.nextSibling)});if(arguments.length){var a=p.clean(arguments);return this.pushStack(p.merge(this,a),"after",this.selector)}},remove:function(a,b){var c,d=0;for(;(c=this[d])!=null;d++)if(!a||p.filter(a,[c]).length)!b&&c.nodeType===1&&(p.cleanData(c.getElementsByTagName("*")),p.cleanData([c])),c.parentNode&&c.parentNode.removeChild(c);return this},empty:function(){var a,b=0;for(;(a=this[b])!=null;b++){a.nodeType===1&&p.cleanData(a.getElementsByTagName("*"));while(a.firstChild)a.removeChild(a.firstChild)}return this},clone:function(a,b){return a=a==null?!1:a,b=b==null?a:b,this.map(function(){return p.clone(this,a,b)})},html:function(a){return p.access(this,function(a){var c=this[0]||{},d=0,e=this.length;if(a===b)return c.nodeType===1?c.innerHTML.replace(bm,""):b;if(typeof a=="string"&&!bs.test(a)&&(p.support.htmlSerialize||!bu.test(a))&&(p.support.leadingWhitespace||!bn.test(a))&&!bz[(bp.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(bo,"<$1>");try{for(;d1&&typeof j=="string"&&bw.test(j))return this.each(function(){p(this).domManip(a,c,d)});if(p.isFunction(j))return this.each(function(e){var f=p(this);a[0]=j.call(this,e,c?f.html():b),f.domManip(a,c,d)});if(this[0]){e=p.buildFragment(a,this,k),g=e.fragment,f=g.firstChild,g.childNodes.length===1&&(g=f);if(f){c=c&&p.nodeName(f,"tr");for(h=e.cacheable||l-1;i0?this.clone(!0):this).get(),p(g[e])[b](d),f=f.concat(d);return this.pushStack(f,a,g.selector)}}),p.extend({clone:function(a,b,c){var d,e,f,g;p.support.html5Clone||p.isXMLDoc(a)||!bu.test("<"+a.nodeName+">")?g=a.cloneNode(!0):(bB.innerHTML=a.outerHTML,bB.removeChild(g=bB.firstChild));if((!p.support.noCloneEvent||!p.support.noCloneChecked)&&(a.nodeType===1||a.nodeType===11)&&!p.isXMLDoc(a)){bE(a,g),d=bF(a),e=bF(g);for(f=0;d[f];++f)e[f]&&bE(d[f],e[f])}if(b){bD(a,g);if(c){d=bF(a),e=bF(g);for(f=0;d[f];++f)bD(d[f],e[f])}}return d=e=null,g},clean:function(a,b,c,d){var f,g,h,i,j,k,l,m,n,o,q,r,s=0,t=[];if(!b||typeof b.createDocumentFragment=="undefined")b=e;for(g=b===e&&bA;(h=a[s])!=null;s++){typeof h=="number"&&(h+="");if(!h)continue;if(typeof h=="string")if(!br.test(h))h=b.createTextNode(h);else{g=g||bk(b),l=l||g.appendChild(b.createElement("div")),h=h.replace(bo,"<$1>"),i=(bp.exec(h)||["",""])[1].toLowerCase(),j=bz[i]||bz._default,k=j[0],l.innerHTML=j[1]+h+j[2];while(k--)l=l.lastChild;if(!p.support.tbody){m=bq.test(h),n=i==="table"&&!m?l.firstChild&&l.firstChild.childNodes:j[1]===""&&!m?l.childNodes:[];for(f=n.length-1;f>=0;--f)p.nodeName(n[f],"tbody")&&!n[f].childNodes.length&&n[f].parentNode.removeChild(n[f])}!p.support.leadingWhitespace&&bn.test(h)&&l.insertBefore(b.createTextNode(bn.exec(h)[0]),l.firstChild),h=l.childNodes,l=g.lastChild}h.nodeType?t.push(h):t=p.merge(t,h)}l&&(g.removeChild(l),h=l=g=null);if(!p.support.appendChecked)for(s=0;(h=t[s])!=null;s++)p.nodeName(h,"input")?bG(h):typeof h.getElementsByTagName!="undefined"&&p.grep(h.getElementsByTagName("input"),bG);if(c){q=function(a){if(!a.type||bx.test(a.type))return d?d.push(a.parentNode?a.parentNode.removeChild(a):a):c.appendChild(a)};for(s=0;(h=t[s])!=null;s++)if(!p.nodeName(h,"script")||!q(h))c.appendChild(h),typeof h.getElementsByTagName!="undefined"&&(r=p.grep(p.merge([],h.getElementsByTagName("script")),q),t.splice.apply(t,[s+1,0].concat(r)),s+=r.length)}return t},cleanData:function(a,b){var c,d,e,f,g=0,h=p.expando,i=p.cache,j=p.support.deleteExpando,k=p.event.special;for(;(e=a[g])!=null;g++)if(b||p.acceptData(e)){d=e[h],c=d&&i[d];if(c){if(c.events)for(f in c.events)k[f]?p.event.remove(e,f):p.removeEvent(e,f,c.handle);i[d]&&(delete i[d],j?delete e[h]:e.removeAttribute?e.removeAttribute(h):e[h]=null,p.deletedIds.push(d))}}}}),function(){var a,b;p.uaMatch=function(a){a=a.toLowerCase();var b=/(chrome)[ \/]([\w.]+)/.exec(a)||/(webkit)[ \/]([\w.]+)/.exec(a)||/(opera)(?:.*version|)[ \/]([\w.]+)/.exec(a)||/(msie) ([\w.]+)/.exec(a)||a.indexOf("compatible")<0&&/(mozilla)(?:.*? rv:([\w.]+)|)/.exec(a)||[];return{browser:b[1]||"",version:b[2]||"0"}},a=p.uaMatch(g.userAgent),b={},a.browser&&(b[a.browser]=!0,b.version=a.version),b.webkit&&(b.safari=!0),p.browser=b,p.sub=function(){function a(b,c){return new a.fn.init(b,c)}p.extend(!0,a,this),a.superclass=this,a.fn=a.prototype=this(),a.fn.constructor=a,a.sub=this.sub,a.fn.init=function c(c,d){return d&&d instanceof p&&!(d instanceof a)&&(d=a(d)),p.fn.init.call(this,c,d,b)},a.fn.init.prototype=a.fn;var b=a(e);return a}}();var bH,bI,bJ,bK=/alpha\([^)]*\)/i,bL=/opacity=([^)]*)/,bM=/^(top|right|bottom|left)$/,bN=/^margin/,bO=new RegExp("^("+q+")(.*)$","i"),bP=new RegExp("^("+q+")(?!px)[a-z%]+$","i"),bQ=new RegExp("^([-+])=("+q+")","i"),bR={},bS={position:"absolute",visibility:"hidden",display:"block"},bT={letterSpacing:0,fontWeight:400,lineHeight:1},bU=["Top","Right","Bottom","Left"],bV=["Webkit","O","Moz","ms"],bW=p.fn.toggle;p.fn.extend({css:function(a,c){return p.access(this,function(a,c,d){return d!==b?p.style(a,c,d):p.css(a,c)},a,c,arguments.length>1)},show:function(){return bZ(this,!0)},hide:function(){return bZ(this)},toggle:function(a,b){var c=typeof a=="boolean";return p.isFunction(a)&&p.isFunction(b)?bW.apply(this,arguments):this.each(function(){(c?a:bY(this))?p(this).show():p(this).hide()})}}),p.extend({cssHooks:{opacity:{get:function(a,b){if(b){var c=bH(a,"opacity");return c===""?"1":c}}}},cssNumber:{fillOpacity:!0,fontWeight:!0,lineHeight:!0,opacity:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":p.support.cssFloat?"cssFloat":"styleFloat"},style:function(a,c,d,e){if(!a||a.nodeType===3||a.nodeType===8||!a.style)return;var f,g,h,i=p.camelCase(c),j=a.style;c=p.cssProps[i]||(p.cssProps[i]=bX(j,i)),h=p.cssHooks[c]||p.cssHooks[i];if(d===b)return h&&"get"in h&&(f=h.get(a,!1,e))!==b?f:j[c];g=typeof d,g==="string"&&(f=bQ.exec(d))&&(d=(f[1]+1)*f[2]+parseFloat(p.css(a,c)),g="number");if(d==null||g==="number"&&isNaN(d))return;g==="number"&&!p.cssNumber[i]&&(d+="px");if(!h||!("set"in h)||(d=h.set(a,d,e))!==b)try{j[c]=d}catch(k){}},css:function(a,c,d,e){var f,g,h,i=p.camelCase(c);return c=p.cssProps[i]||(p.cssProps[i]=bX(a.style,i)),h=p.cssHooks[c]||p.cssHooks[i],h&&"get"in h&&(f=h.get(a,!0,e)),f===b&&(f=bH(a,c)),f==="normal"&&c in bT&&(f=bT[c]),d||e!==b?(g=parseFloat(f),d||p.isNumeric(g)?g||0:f):f},swap:function(a,b,c){var d,e,f={};for(e in b)f[e]=a.style[e],a.style[e]=b[e];d=c.call(a);for(e in b)a.style[e]=f[e];return d}}),a.getComputedStyle?bH=function(a,b){var c,d,e,f,g=getComputedStyle(a,null),h=a.style;return g&&(c=g[b],c===""&&!p.contains(a.ownerDocument.documentElement,a)&&(c=p.style(a,b)),bP.test(c)&&bN.test(b)&&(d=h.width,e=h.minWidth,f=h.maxWidth,h.minWidth=h.maxWidth=h.width=c,c=g.width,h.width=d,h.minWidth=e,h.maxWidth=f)),c}:e.documentElement.currentStyle&&(bH=function(a,b){var c,d,e=a.currentStyle&&a.currentStyle[b],f=a.style;return e==null&&f&&f[b]&&(e=f[b]),bP.test(e)&&!bM.test(b)&&(c=f.left,d=a.runtimeStyle&&a.runtimeStyle.left,d&&(a.runtimeStyle.left=a.currentStyle.left),f.left=b==="fontSize"?"1em":e,e=f.pixelLeft+"px",f.left=c,d&&(a.runtimeStyle.left=d)),e===""?"auto":e}),p.each(["height","width"],function(a,b){p.cssHooks[b]={get:function(a,c,d){if(c)return a.offsetWidth!==0||bH(a,"display")!=="none"?ca(a,b,d):p.swap(a,bS,function(){return ca(a,b,d)})},set:function(a,c,d){return b$(a,c,d?b_(a,b,d,p.support.boxSizing&&p.css(a,"boxSizing")==="border-box"):0)}}}),p.support.opacity||(p.cssHooks.opacity={get:function(a,b){return bL.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?.01*parseFloat(RegExp.$1)+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle,e=p.isNumeric(b)?"alpha(opacity="+b*100+")":"",f=d&&d.filter||c.filter||"";c.zoom=1;if(b>=1&&p.trim(f.replace(bK,""))===""&&c.removeAttribute){c.removeAttribute("filter");if(d&&!d.filter)return}c.filter=bK.test(f)?f.replace(bK,e):f+" "+e}}),p(function(){p.support.reliableMarginRight||(p.cssHooks.marginRight={get:function(a,b){return p.swap(a,{display:"inline-block"},function(){if(b)return bH(a,"marginRight")})}}),!p.support.pixelPosition&&p.fn.position&&p.each(["top","left"],function(a,b){p.cssHooks[b]={get:function(a,c){if(c){var d=bH(a,b);return bP.test(d)?p(a).position()[b]+"px":d}}}})}),p.expr&&p.expr.filters&&(p.expr.filters.hidden=function(a){return a.offsetWidth===0&&a.offsetHeight===0||!p.support.reliableHiddenOffsets&&(a.style&&a.style.display||bH(a,"display"))==="none"},p.expr.filters.visible=function(a){return!p.expr.filters.hidden(a)}),p.each({margin:"",padding:"",border:"Width"},function(a,b){p.cssHooks[a+b]={expand:function(c){var d,e=typeof c=="string"?c.split(" "):[c],f={};for(d=0;d<4;d++)f[a+bU[d]+b]=e[d]||e[d-2]||e[0];return f}},bN.test(a)||(p.cssHooks[a+b].set=b$)});var cc=/%20/g,cd=/\[\]$/,ce=/\r?\n/g,cf=/^(?:color|date|datetime|datetime-local|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,cg=/^(?:select|textarea)/i;p.fn.extend({serialize:function(){return p.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?p.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||cg.test(this.nodeName)||cf.test(this.type))}).map(function(a,b){var c=p(this).val();return c==null?null:p.isArray(c)?p.map(c,function(a,c){return{name:b.name,value:a.replace(ce,"\r\n")}}):{name:b.name,value:c.replace(ce,"\r\n")}}).get()}}),p.param=function(a,c){var d,e=[],f=function(a,b){b=p.isFunction(b)?b():b==null?"":b,e[e.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};c===b&&(c=p.ajaxSettings&&p.ajaxSettings.traditional);if(p.isArray(a)||a.jquery&&!p.isPlainObject(a))p.each(a,function(){f(this.name,this.value)});else for(d in a)ch(d,a[d],c,f);return e.join("&").replace(cc,"+")};var ci,cj,ck=/#.*$/,cl=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,cm=/^(?:about|app|app\-storage|.+\-extension|file|res|widget):$/,cn=/^(?:GET|HEAD)$/,co=/^\/\//,cp=/\?/,cq=/)<[^<]*)*<\/script>/gi,cr=/([?&])_=[^&]*/,cs=/^([\w\+\.\-]+:)(?:\/\/([^\/?#:]*)(?::(\d+)|)|)/,ct=p.fn.load,cu={},cv={},cw=["*/"]+["*"];try{ci=f.href}catch(cx){ci=e.createElement("a"),ci.href="",ci=ci.href}cj=cs.exec(ci.toLowerCase())||[],p.fn.load=function(a,c,d){if(typeof a!="string"&&ct)return ct.apply(this,arguments);if(!this.length)return this;var e,f,g,h=this,i=a.indexOf(" ");return i>=0&&(e=a.slice(i,a.length),a=a.slice(0,i)),p.isFunction(c)?(d=c,c=b):typeof c=="object"&&(f="POST"),p.ajax({url:a,type:f,dataType:"html",data:c,complete:function(a,b){d&&h.each(d,g||[a.responseText,b,a])}}).done(function(a){g=arguments,h.html(e?p("
").append(a.replace(cq,"")).find(e):a)}),this},p.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "),function(a,b){p.fn[b]=function(a){return this.on(b,a)}}),p.each(["get","post"],function(a,c){p[c]=function(a,d,e,f){return p.isFunction(d)&&(f=f||e,e=d,d=b),p.ajax({type:c,url:a,data:d,success:e,dataType:f})}}),p.extend({getScript:function(a,c){return p.get(a,b,c,"script")},getJSON:function(a,b,c){return p.get(a,b,c,"json")},ajaxSetup:function(a,b){return b?cA(a,p.ajaxSettings):(b=a,a=p.ajaxSettings),cA(a,b),a},ajaxSettings:{url:ci,isLocal:cm.test(cj[1]),global:!0,type:"GET",contentType:"application/x-www-form-urlencoded; charset=UTF-8",processData:!0,async:!0,accepts:{xml:"application/xml, text/xml",html:"text/html",text:"text/plain",json:"application/json, text/javascript","*":cw},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText"},converters:{"* text":a.String,"text html":!0,"text json":p.parseJSON,"text xml":p.parseXML},flatOptions:{context:!0,url:!0}},ajaxPrefilter:cy(cu),ajaxTransport:cy(cv),ajax:function(a,c){function y(a,c,f,i){var k,s,t,u,w,y=c;if(v===2)return;v=2,h&&clearTimeout(h),g=b,e=i||"",x.readyState=a>0?4:0,f&&(u=cB(l,x,f));if(a>=200&&a<300||a===304)l.ifModified&&(w=x.getResponseHeader("Last-Modified"),w&&(p.lastModified[d]=w),w=x.getResponseHeader("Etag"),w&&(p.etag[d]=w)),a===304?(y="notmodified",k=!0):(k=cC(l,u),y=k.state,s=k.data,t=k.error,k=!t);else{t=y;if(!y||a)y="error",a<0&&(a=0)}x.status=a,x.statusText=""+(c||y),k?o.resolveWith(m,[s,y,x]):o.rejectWith(m,[x,y,t]),x.statusCode(r),r=b,j&&n.trigger("ajax"+(k?"Success":"Error"),[x,l,k?s:t]),q.fireWith(m,[x,y]),j&&(n.trigger("ajaxComplete",[x,l]),--p.active||p.event.trigger("ajaxStop"))}typeof a=="object"&&(c=a,a=b),c=c||{};var d,e,f,g,h,i,j,k,l=p.ajaxSetup({},c),m=l.context||l,n=m!==l&&(m.nodeType||m instanceof p)?p(m):p.event,o=p.Deferred(),q=p.Callbacks("once memory"),r=l.statusCode||{},t={},u={},v=0,w="canceled",x={readyState:0,setRequestHeader:function(a,b){if(!v){var c=a.toLowerCase();a=u[c]=u[c]||a,t[a]=b}return this},getAllResponseHeaders:function(){return v===2?e:null},getResponseHeader:function(a){var c;if(v===2){if(!f){f={};while(c=cl.exec(e))f[c[1].toLowerCase()]=c[2]}c=f[a.toLowerCase()]}return c===b?null:c},overrideMimeType:function(a){return v||(l.mimeType=a),this},abort:function(a){return a=a||w,g&&g.abort(a),y(0,a),this}};o.promise(x),x.success=x.done,x.error=x.fail,x.complete=q.add,x.statusCode=function(a){if(a){var b;if(v<2)for(b in a)r[b]=[r[b],a[b]];else b=a[x.status],x.always(b)}return this},l.url=((a||l.url)+"").replace(ck,"").replace(co,cj[1]+"//"),l.dataTypes=p.trim(l.dataType||"*").toLowerCase().split(s),l.crossDomain==null&&(i=cs.exec(l.url.toLowerCase()),l.crossDomain=!(!i||i[1]==cj[1]&&i[2]==cj[2]&&(i[3]||(i[1]==="http:"?80:443))==(cj[3]||(cj[1]==="http:"?80:443)))),l.data&&l.processData&&typeof l.data!="string"&&(l.data=p.param(l.data,l.traditional)),cz(cu,l,c,x);if(v===2)return x;j=l.global,l.type=l.type.toUpperCase(),l.hasContent=!cn.test(l.type),j&&p.active++===0&&p.event.trigger("ajaxStart");if(!l.hasContent){l.data&&(l.url+=(cp.test(l.url)?"&":"?")+l.data,delete l.data),d=l.url;if(l.cache===!1){var z=p.now(),A=l.url.replace(cr,"$1_="+z);l.url=A+(A===l.url?(cp.test(l.url)?"&":"?")+"_="+z:"")}}(l.data&&l.hasContent&&l.contentType!==!1||c.contentType)&&x.setRequestHeader("Content-Type",l.contentType),l.ifModified&&(d=d||l.url,p.lastModified[d]&&x.setRequestHeader("If-Modified-Since",p.lastModified[d]),p.etag[d]&&x.setRequestHeader("If-None-Match",p.etag[d])),x.setRequestHeader("Accept",l.dataTypes[0]&&l.accepts[l.dataTypes[0]]?l.accepts[l.dataTypes[0]]+(l.dataTypes[0]!=="*"?", "+cw+"; q=0.01":""):l.accepts["*"]);for(k in l.headers)x.setRequestHeader(k,l.headers[k]);if(!l.beforeSend||l.beforeSend.call(m,x,l)!==!1&&v!==2){w="abort";for(k in{success:1,error:1,complete:1})x[k](l[k]);g=cz(cv,l,c,x);if(!g)y(-1,"No Transport");else{x.readyState=1,j&&n.trigger("ajaxSend",[x,l]),l.async&&l.timeout>0&&(h=setTimeout(function(){x.abort("timeout")},l.timeout));try{v=1,g.send(t,y)}catch(B){if(v<2)y(-1,B);else throw B}}return x}return x.abort()},active:0,lastModified:{},etag:{}});var cD=[],cE=/\?/,cF=/(=)\?(?=&|$)|\?\?/,cG=p.now();p.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var a=cD.pop()||p.expando+"_"+cG++;return this[a]=!0,a}}),p.ajaxPrefilter("json jsonp",function(c,d,e){var f,g,h,i=c.data,j=c.url,k=c.jsonp!==!1,l=k&&cF.test(j),m=k&&!l&&typeof i=="string"&&!(c.contentType||"").indexOf("application/x-www-form-urlencoded")&&cF.test(i);if(c.dataTypes[0]==="jsonp"||l||m)return f=c.jsonpCallback=p.isFunction(c.jsonpCallback)?c.jsonpCallback():c.jsonpCallback,g=a[f],l?c.url=j.replace(cF,"$1"+f):m?c.data=i.replace(cF,"$1"+f):k&&(c.url+=(cE.test(j)?"&":"?")+c.jsonp+"="+f),c.converters["script json"]=function(){return h||p.error(f+" was not called"),h[0]},c.dataTypes[0]="json",a[f]=function(){h=arguments},e.always(function(){a[f]=g,c[f]&&(c.jsonpCallback=d.jsonpCallback,cD.push(f)),h&&p.isFunction(g)&&g(h[0]),h=g=b}),"script"}),p.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/javascript|ecmascript/},converters:{"text script":function(a){return p.globalEval(a),a}}}),p.ajaxPrefilter("script",function(a){a.cache===b&&(a.cache=!1),a.crossDomain&&(a.type="GET",a.global=!1)}),p.ajaxTransport("script",function(a){if(a.crossDomain){var c,d=e.head||e.getElementsByTagName("head")[0]||e.documentElement;return{send:function(f,g){c=e.createElement("script"),c.async="async",a.scriptCharset&&(c.charset=a.scriptCharset),c.src=a.url,c.onload=c.onreadystatechange=function(a,e){if(e||!c.readyState||/loaded|complete/.test(c.readyState))c.onload=c.onreadystatechange=null,d&&c.parentNode&&d.removeChild(c),c=b,e||g(200,"success")},d.insertBefore(c,d.firstChild)},abort:function(){c&&c.onload(0,1)}}}});var cH,cI=a.ActiveXObject?function(){for(var a in cH)cH[a](0,1)}:!1,cJ=0;p.ajaxSettings.xhr=a.ActiveXObject?function(){return!this.isLocal&&cK()||cL()}:cK,function(a){p.extend(p.support,{ajax:!!a,cors:!!a&&"withCredentials"in a})}(p.ajaxSettings.xhr()),p.support.ajax&&p.ajaxTransport(function(c){if(!c.crossDomain||p.support.cors){var d;return{send:function(e,f){var g,h,i=c.xhr();c.username?i.open(c.type,c.url,c.async,c.username,c.password):i.open(c.type,c.url,c.async);if(c.xhrFields)for(h in c.xhrFields)i[h]=c.xhrFields[h];c.mimeType&&i.overrideMimeType&&i.overrideMimeType(c.mimeType),!c.crossDomain&&!e["X-Requested-With"]&&(e["X-Requested-With"]="XMLHttpRequest");try{for(h in e)i.setRequestHeader(h,e[h])}catch(j){}i.send(c.hasContent&&c.data||null),d=function(a,e){var h,j,k,l,m;try{if(d&&(e||i.readyState===4)){d=b,g&&(i.onreadystatechange=p.noop,cI&&delete cH[g]);if(e)i.readyState!==4&&i.abort();else{h=i.status,k=i.getAllResponseHeaders(),l={},m=i.responseXML,m&&m.documentElement&&(l.xml=m);try{l.text=i.responseText}catch(a){}try{j=i.statusText}catch(n){j=""}!h&&c.isLocal&&!c.crossDomain?h=l.text?200:404:h===1223&&(h=204)}}}catch(o){e||f(-1,o)}l&&f(h,j,l,k)},c.async?i.readyState===4?setTimeout(d,0):(g=++cJ,cI&&(cH||(cH={},p(a).unload(cI)),cH[g]=d),i.onreadystatechange=d):d()},abort:function(){d&&d(0,1)}}}});var cM,cN,cO=/^(?:toggle|show|hide)$/,cP=new RegExp("^(?:([-+])=|)("+q+")([a-z%]*)$","i"),cQ=/queueHooks$/,cR=[cX],cS={"*":[function(a,b){var c,d,e,f=this.createTween(a,b),g=cP.exec(b),h=f.cur(),i=+h||0,j=1;if(g){c=+g[2],d=g[3]||(p.cssNumber[a]?"":"px");if(d!=="px"&&i){i=p.css(f.elem,a,!0)||c||1;do e=j=j||".5",i=i/j,p.style(f.elem,a,i+d),j=f.cur()/h;while(j!==1&&j!==e)}f.unit=d,f.start=i,f.end=g[1]?i+(g[1]+1)*c:c}return f}]};p.Animation=p.extend(cV,{tweener:function(a,b){p.isFunction(a)?(b=a,a=["*"]):a=a.split(" ");var c,d=0,e=a.length;for(;d-1,j={},k={},l,m;i?(k=e.position(),l=k.top,m=k.left):(l=parseFloat(g)||0,m=parseFloat(h)||0),p.isFunction(b)&&(b=b.call(a,c,f)),b.top!=null&&(j.top=b.top-f.top+l),b.left!=null&&(j.left=b.left-f.left+m),"using"in b?b.using.call(a,j):e.css(j)}},p.fn.extend({position:function(){if(!this[0])return;var a=this[0],b=this.offsetParent(),c=this.offset(),d=c$.test(b[0].nodeName)?{top:0,left:0}:b.offset();return c.top-=parseFloat(p.css(a,"marginTop"))||0,c.left-=parseFloat(p.css(a,"marginLeft"))||0,d.top+=parseFloat(p.css(b[0],"borderTopWidth"))||0,d.left+=parseFloat(p.css(b[0],"borderLeftWidth"))||0,{top:c.top-d.top,left:c.left-d.left}},offsetParent:function(){return this.map(function(){var a=this.offsetParent||e.body;while(a&&!c$.test(a.nodeName)&&p.css(a,"position")==="static")a=a.offsetParent;return a||e.body})}}),p.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(a,c){var d=/Y/.test(c);p.fn[a]=function(e){return p.access(this,function(a,e,f){var g=c_(a);if(f===b)return g?c in g?g[c]:g.document.documentElement[e]:a[e];g?g.scrollTo(d?p(g).scrollLeft():f,d?f:p(g).scrollTop()):a[e]=f},a,e,arguments.length,null)}}),p.each({Height:"height",Width:"width"},function(a,c){p.each({padding:"inner"+a,content:c,"":"outer"+a},function(d,e){p.fn[e]=function(e,f){var g=arguments.length&&(d||typeof e!="boolean"),h=d||(e===!0||f===!0?"margin":"border");return p.access(this,function(c,d,e){var f;return p.isWindow(c)?c.document.documentElement["client"+a]:c.nodeType===9?(f=c.documentElement,Math.max(c.body["scroll"+a],f["scroll"+a],c.body["offset"+a],f["offset"+a],f["client"+a])):e===b?p.css(c,d,e,h):p.style(c,d,e,h)},c,g?e:b,g)}})}),a.jQuery=a.$=p,typeof define=="function"&&define.amd&&define.amd.jQuery&&define("jquery",[],function(){return p})})(window);PKH1H%@@%iprestrict/fixtures/default_data.json[ { "pk": 1, "model": "iprestrict.ipgroup", "fields": { "name": "ALL", "description": "Matches all IP Addresses" } }, { "pk": 2, "model": "iprestrict.ipgroup", "fields": { "name": "localhost", "description": "IP address of localhost" } }, { "pk": 1, "model": "iprestrict.iprange", "fields": { "first_ip": "0.0.0.0", "ip_group": 1, "last_ip": "255.255.255.255" } }, { "pk": 2, "model": "iprestrict.iprange", "fields": { "first_ip": "0:0:0:0:0:0:0:0", "ip_group": 1, "last_ip": "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff" } }, { "pk": 3, "model": "iprestrict.iprange", "fields": { "last_ip": null, "first_ip": "127.0.0.1", "ip_group": 2 } }, { "pk": 4, "model": "iprestrict.iprange", "fields": { "last_ip": null, "first_ip": "::1", "ip_group": 2 } }, { "pk": 1, "model": "iprestrict.rule", "fields": { "action": "A", "url_pattern": "ALL", "rank": 65535, "ip_group": 2 } }, { "pk": 2, "model": "iprestrict.rule", "fields": { "action": "D", "url_pattern": "ALL", "rank": 65536, "ip_group": 1 } } ] PKt[C'r--%iprestrict/migrations/0001_initial.py# encoding: utf-8 import datetime from south.db import db from south.v2 import SchemaMigration from django.db import models class Migration(SchemaMigration): def forwards(self, orm): # Adding model 'IPGroup' db.create_table('iprestrict_ipgroup', ( ('id', self.gf('django.db.models.fields.AutoField')(primary_key=True)), ('name', self.gf('django.db.models.fields.CharField')(max_length=100)), ('description', self.gf('django.db.models.fields.TextField')(null=True, blank=True)), )) db.send_create_signal('iprestrict', ['IPGroup']) # Adding model 'IPRange' db.create_table('iprestrict_iprange', ( ('id', self.gf('django.db.models.fields.AutoField')(primary_key=True)), ('ip_group', self.gf('django.db.models.fields.related.ForeignKey')(to=orm['iprestrict.IPGroup'])), ('first_ip', self.gf('django.db.models.fields.GenericIPAddressField')(max_length=39)), ('cidr_prefix_length', self.gf('django.db.models.fields.PositiveSmallIntegerField')(null=True, blank=True)), ('last_ip', self.gf('django.db.models.fields.GenericIPAddressField')(max_length=39, null=True, blank=True)), )) db.send_create_signal('iprestrict', ['IPRange']) # Adding model 'Rule' db.create_table('iprestrict_rule', ( ('id', self.gf('django.db.models.fields.AutoField')(primary_key=True)), ('url_pattern', self.gf('django.db.models.fields.CharField')(max_length=500)), ('ip_group', self.gf('django.db.models.fields.related.ForeignKey')(default=1, to=orm['iprestrict.IPGroup'])), ('action', self.gf('django.db.models.fields.CharField')(default='D', max_length=1)), ('rank', self.gf('django.db.models.fields.IntegerField')(blank=True)), )) db.send_create_signal('iprestrict', ['Rule']) # Adding model 'ReloadRulesRequest' db.create_table('iprestrict_reloadrulesrequest', ( ('id', self.gf('django.db.models.fields.AutoField')(primary_key=True)), ('at', self.gf('django.db.models.fields.DateTimeField')(auto_now_add=True, blank=True)), )) db.send_create_signal('iprestrict', ['ReloadRulesRequest']) def backwards(self, orm): # Deleting model 'IPGroup' db.delete_table('iprestrict_ipgroup') # Deleting model 'IPRange' db.delete_table('iprestrict_iprange') # Deleting model 'Rule' db.delete_table('iprestrict_rule') # Deleting model 'ReloadRulesRequest' db.delete_table('iprestrict_reloadrulesrequest') models = { 'iprestrict.ipgroup': { 'Meta': {'object_name': 'IPGroup'}, 'description': ('django.db.models.fields.TextField', [], {'null': 'True', 'blank': 'True'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'name': ('django.db.models.fields.CharField', [], {'max_length': '100'}) }, 'iprestrict.iprange': { 'Meta': {'object_name': 'IPRange'}, 'cidr_prefix_length': ('django.db.models.fields.PositiveSmallIntegerField', [], {'null': 'True', 'blank': 'True'}), 'first_ip': ('django.db.models.fields.GenericIPAddressField', [], {'max_length': '39'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'ip_group': ('django.db.models.fields.related.ForeignKey', [], {'to': "orm['iprestrict.IPGroup']"}), 'last_ip': ('django.db.models.fields.GenericIPAddressField', [], {'max_length': '39', 'null': 'True', 'blank': 'True'}) }, 'iprestrict.reloadrulesrequest': { 'Meta': {'object_name': 'ReloadRulesRequest'}, 'at': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}) }, 'iprestrict.rule': { 'Meta': {'ordering': "['rank', 'id']", 'object_name': 'Rule'}, 'action': ('django.db.models.fields.CharField', [], {'default': "'D'", 'max_length': '1'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'ip_group': ('django.db.models.fields.related.ForeignKey', [], {'default': '1', 'to': "orm['iprestrict.IPGroup']"}), 'rank': ('django.db.models.fields.IntegerField', [], {'blank': 'True'}), 'url_pattern': ('django.db.models.fields.CharField', [], {'max_length': '500'}) } } complete_apps = ['iprestrict'] PKt[C㨺E E 1iprestrict/migrations/0002_create_initial_data.py# encoding: utf-8 import datetime from south.db import db from south.v2 import DataMigration from django.db import models class Migration(DataMigration): def forwards(self, orm): # Default IP Groups: ALL and localhost all_group = orm.IPGroup.objects.create( name='ALL', description="Matches all IP Addresses") localhost_group = orm.IPGroup.objects.create( name='localhost', description="IP Address of localhost") # IP ranges defining ALL and localhost for ipv4 and ipv6 orm.IPRange.objects.create(ip_group=all_group, first_ip = '0.0.0.0', last_ip = '255.255.255.255') orm.IPRange.objects.create(ip_group=all_group, first_ip = '0:0:0:0:0:0:0:0', last_ip = 'ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff') orm.IPRange.objects.create(ip_group=localhost_group, first_ip = '127.0.0.1') orm.IPRange.objects.create(ip_group=all_group, first_ip = '::1') # Default rules: Allow all for localhost and Deny everything else orm.Rule.objects.create(ip_group=localhost_group, action = 'A', url_pattern = 'ALL', rank=65535) orm.Rule.objects.create(ip_group=all_group, action = 'D', url_pattern = 'ALL', rank=65536) def backwards(self, orm): raise RuntimeError("Cannot reverse this migration.") models = { 'iprestrict.ipgroup': { 'Meta': {'object_name': 'IPGroup'}, 'description': ('django.db.models.fields.TextField', [], {'null': 'True', 'blank': 'True'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'name': ('django.db.models.fields.CharField', [], {'max_length': '100'}) }, 'iprestrict.iprange': { 'Meta': {'object_name': 'IPRange'}, 'cidr_prefix_length': ('django.db.models.fields.PositiveSmallIntegerField', [], {'null': 'True', 'blank': 'True'}), 'first_ip': ('django.db.models.fields.GenericIPAddressField', [], {'max_length': '39'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'ip_group': ('django.db.models.fields.related.ForeignKey', [], {'to': "orm['iprestrict.IPGroup']"}), 'last_ip': ('django.db.models.fields.GenericIPAddressField', [], {'max_length': '39', 'null': 'True', 'blank': 'True'}) }, 'iprestrict.reloadrulesrequest': { 'Meta': {'object_name': 'ReloadRulesRequest'}, 'at': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}) }, 'iprestrict.rule': { 'Meta': {'ordering': "['rank', 'id']", 'object_name': 'Rule'}, 'action': ('django.db.models.fields.CharField', [], {'default': "'D'", 'max_length': '1'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'ip_group': ('django.db.models.fields.related.ForeignKey', [], {'default': '1', 'to': "orm['iprestrict.IPGroup']"}), 'rank': ('django.db.models.fields.IntegerField', [], {'blank': 'True'}), 'url_pattern': ('django.db.models.fields.CharField', [], {'max_length': '500'}) } } complete_apps = ['iprestrict'] PKt[C!iprestrict/migrations/__init__.pyPKH1H'r--+iprestrict/south_migrations/0001_initial.py# encoding: utf-8 import datetime from south.db import db from south.v2 import SchemaMigration from django.db import models class Migration(SchemaMigration): def forwards(self, orm): # Adding model 'IPGroup' db.create_table('iprestrict_ipgroup', ( ('id', self.gf('django.db.models.fields.AutoField')(primary_key=True)), ('name', self.gf('django.db.models.fields.CharField')(max_length=100)), ('description', self.gf('django.db.models.fields.TextField')(null=True, blank=True)), )) db.send_create_signal('iprestrict', ['IPGroup']) # Adding model 'IPRange' db.create_table('iprestrict_iprange', ( ('id', self.gf('django.db.models.fields.AutoField')(primary_key=True)), ('ip_group', self.gf('django.db.models.fields.related.ForeignKey')(to=orm['iprestrict.IPGroup'])), ('first_ip', self.gf('django.db.models.fields.GenericIPAddressField')(max_length=39)), ('cidr_prefix_length', self.gf('django.db.models.fields.PositiveSmallIntegerField')(null=True, blank=True)), ('last_ip', self.gf('django.db.models.fields.GenericIPAddressField')(max_length=39, null=True, blank=True)), )) db.send_create_signal('iprestrict', ['IPRange']) # Adding model 'Rule' db.create_table('iprestrict_rule', ( ('id', self.gf('django.db.models.fields.AutoField')(primary_key=True)), ('url_pattern', self.gf('django.db.models.fields.CharField')(max_length=500)), ('ip_group', self.gf('django.db.models.fields.related.ForeignKey')(default=1, to=orm['iprestrict.IPGroup'])), ('action', self.gf('django.db.models.fields.CharField')(default='D', max_length=1)), ('rank', self.gf('django.db.models.fields.IntegerField')(blank=True)), )) db.send_create_signal('iprestrict', ['Rule']) # Adding model 'ReloadRulesRequest' db.create_table('iprestrict_reloadrulesrequest', ( ('id', self.gf('django.db.models.fields.AutoField')(primary_key=True)), ('at', self.gf('django.db.models.fields.DateTimeField')(auto_now_add=True, blank=True)), )) db.send_create_signal('iprestrict', ['ReloadRulesRequest']) def backwards(self, orm): # Deleting model 'IPGroup' db.delete_table('iprestrict_ipgroup') # Deleting model 'IPRange' db.delete_table('iprestrict_iprange') # Deleting model 'Rule' db.delete_table('iprestrict_rule') # Deleting model 'ReloadRulesRequest' db.delete_table('iprestrict_reloadrulesrequest') models = { 'iprestrict.ipgroup': { 'Meta': {'object_name': 'IPGroup'}, 'description': ('django.db.models.fields.TextField', [], {'null': 'True', 'blank': 'True'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'name': ('django.db.models.fields.CharField', [], {'max_length': '100'}) }, 'iprestrict.iprange': { 'Meta': {'object_name': 'IPRange'}, 'cidr_prefix_length': ('django.db.models.fields.PositiveSmallIntegerField', [], {'null': 'True', 'blank': 'True'}), 'first_ip': ('django.db.models.fields.GenericIPAddressField', [], {'max_length': '39'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'ip_group': ('django.db.models.fields.related.ForeignKey', [], {'to': "orm['iprestrict.IPGroup']"}), 'last_ip': ('django.db.models.fields.GenericIPAddressField', [], {'max_length': '39', 'null': 'True', 'blank': 'True'}) }, 'iprestrict.reloadrulesrequest': { 'Meta': {'object_name': 'ReloadRulesRequest'}, 'at': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}) }, 'iprestrict.rule': { 'Meta': {'ordering': "['rank', 'id']", 'object_name': 'Rule'}, 'action': ('django.db.models.fields.CharField', [], {'default': "'D'", 'max_length': '1'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'ip_group': ('django.db.models.fields.related.ForeignKey', [], {'default': '1', 'to': "orm['iprestrict.IPGroup']"}), 'rank': ('django.db.models.fields.IntegerField', [], {'blank': 'True'}), 'url_pattern': ('django.db.models.fields.CharField', [], {'max_length': '500'}) } } complete_apps = ['iprestrict'] PKH1H㨺E E 7iprestrict/south_migrations/0002_create_initial_data.py# encoding: utf-8 import datetime from south.db import db from south.v2 import DataMigration from django.db import models class Migration(DataMigration): def forwards(self, orm): # Default IP Groups: ALL and localhost all_group = orm.IPGroup.objects.create( name='ALL', description="Matches all IP Addresses") localhost_group = orm.IPGroup.objects.create( name='localhost', description="IP Address of localhost") # IP ranges defining ALL and localhost for ipv4 and ipv6 orm.IPRange.objects.create(ip_group=all_group, first_ip = '0.0.0.0', last_ip = '255.255.255.255') orm.IPRange.objects.create(ip_group=all_group, first_ip = '0:0:0:0:0:0:0:0', last_ip = 'ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff') orm.IPRange.objects.create(ip_group=localhost_group, first_ip = '127.0.0.1') orm.IPRange.objects.create(ip_group=all_group, first_ip = '::1') # Default rules: Allow all for localhost and Deny everything else orm.Rule.objects.create(ip_group=localhost_group, action = 'A', url_pattern = 'ALL', rank=65535) orm.Rule.objects.create(ip_group=all_group, action = 'D', url_pattern = 'ALL', rank=65536) def backwards(self, orm): raise RuntimeError("Cannot reverse this migration.") models = { 'iprestrict.ipgroup': { 'Meta': {'object_name': 'IPGroup'}, 'description': ('django.db.models.fields.TextField', [], {'null': 'True', 'blank': 'True'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'name': ('django.db.models.fields.CharField', [], {'max_length': '100'}) }, 'iprestrict.iprange': { 'Meta': {'object_name': 'IPRange'}, 'cidr_prefix_length': ('django.db.models.fields.PositiveSmallIntegerField', [], {'null': 'True', 'blank': 'True'}), 'first_ip': ('django.db.models.fields.GenericIPAddressField', [], {'max_length': '39'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'ip_group': ('django.db.models.fields.related.ForeignKey', [], {'to': "orm['iprestrict.IPGroup']"}), 'last_ip': ('django.db.models.fields.GenericIPAddressField', [], {'max_length': '39', 'null': 'True', 'blank': 'True'}) }, 'iprestrict.reloadrulesrequest': { 'Meta': {'object_name': 'ReloadRulesRequest'}, 'at': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}) }, 'iprestrict.rule': { 'Meta': {'ordering': "['rank', 'id']", 'object_name': 'Rule'}, 'action': ('django.db.models.fields.CharField', [], {'default': "'D'", 'max_length': '1'}), 'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}), 'ip_group': ('django.db.models.fields.related.ForeignKey', [], {'default': '1', 'to': "orm['iprestrict.IPGroup']"}), 'rank': ('django.db.models.fields.IntegerField', [], {'blank': 'True'}), 'url_pattern': ('django.db.models.fields.CharField', [], {'max_length': '500'}) } } complete_apps = ['iprestrict'] PKH1H'iprestrict/south_migrations/__init__.pyPKH1H2iprestrict/tests/__init__.py PKH1H{k!iprestrict/tests/test_ip_utils.pyfrom django.utils.unittest import TestCase, skip from iprestrict import ip_utils as ipu class IPToNumberTest(TestCase): def test_ip_to_number_conversions(self): self.assertEqual(0, ipu.to_number('0.0.0.0')) self.assertEqual(1, ipu.to_number('0.0.0.1')) self.assertEqual(256, ipu.to_number('0.0.1.0')) self.assertEqual(65537, ipu.to_number('0.1.0.1')) self.assertEqual(16842753, ipu.to_number('1.1.0.1')) @skip('IPv6 to number conversion not ready') def test_ip_to_number_conversions_ipv6(self): self.assertEqual(0, ipu_to_number('::')) class NumberToIPTest(TestCase): def test_number_to_ip_conversions(self): self.assertEqual('0.0.0.0', ipu.to_ip(0)) self.assertEqual('0.0.0.1', ipu.to_ip(1)) self.assertEqual('0.0.1.0', ipu.to_ip(256)) self.assertEqual('0.1.0.1', ipu.to_ip(65537)) self.assertEqual('1.1.0.1', ipu.to_ip(16842753)) PKH1H X!s#iprestrict/tests/test_middleware.pyfrom django.test import TestCase from django.test.client import RequestFactory from django.test.utils import override_settings from django.core import exceptions from iprestrict import models from iprestrict import restrictor from iprestrict.middleware import IPRestrictMiddleware from datetime import datetime LOCAL_IP = '192.168.1.1' PROXY = '1.1.1.1' class MiddlewareRestrictsTest(TestCase): ''' When the middleware is enabled it should restrict all IPs(but localhost)/URLs by default. ''' def setUp(self): models.ReloadRulesRequest.request_reload() def assert_url_is_restricted(self, url): response = self.client.get(url, REMOTE_ADDR = LOCAL_IP) self.assertEqual(response.status_code, 403) def assert_ip_is_restricted(self, ip): response = self.client.get('', REMOTE_ADDR = ip) self.assertEqual(response.status_code, 403) def test_middleware_restricts_every_url(self): self.assert_url_is_restricted('') self.assert_url_is_restricted('/every') self.assert_url_is_restricted('/url') self.assert_url_is_restricted('/is_restricted') self.assert_url_is_restricted('/every/url/is_restricted') def test_middleware_restricts_ips(self): self.assert_ip_is_restricted('192.168.1.1') self.assert_ip_is_restricted('10.10.10.1') self.assert_ip_is_restricted('169.254.0.1') def test_middleware_allows_localhost(self): response = self.client.get('/some/url', REMOTE_ADDR = '127.0.0.1') self.assertEqual(response.status_code, 404) def create_ip_allow_rule(ip=LOCAL_IP): localip = models.IPGroup.objects.create(name='localip') models.IPRange.objects.create(ip_group=localip, first_ip=LOCAL_IP) models.Rule.objects.create(url_pattern='ALL', ip_group = localip, action='A') class MiddlewareAllowsTest(TestCase): def setUp(self): create_ip_allow_rule() models.ReloadRulesRequest.request_reload() def test_middleware_allows_localhost(self): response = self.client.get('') self.assertEqual(response.status_code, 404) def test_middleware_allows_ip_just_added(self): response = self.client.get('', REMOTE_ADDR = LOCAL_IP) self.assertEqual(response.status_code, 404) def test_middleware_restricts_other_ip(self): response = self.client.get('', REMOTE_ADDR = '10.1.1.1') self.assertEqual(response.status_code, 403) @override_settings(TRUSTED_PROXIES=(PROXY,)) def test_middleware_allows_if_proxy_is_trusted(self): response = self.client.get('', REMOTE_ADDR = PROXY, HTTP_X_FORWARDED_FOR= LOCAL_IP) self.assertEqual(response.status_code, 404) def test_middleware_restricts_if_proxy_is_not_trusted(self): response = self.client.get('', REMOTE_ADDR = PROXY, HTTP_X_FORWARDED_FOR = LOCAL_IP) self.assertEqual(response.status_code, 403) class ReloadRulesTest(TestCase): def setUp(self): create_ip_allow_rule() def test_reload_with_custom_command(self): from django.core.management import call_command call_command('reloadrules', verbosity=0) response = self.client.get('', REMOTE_ADDR = LOCAL_IP) self.assertEqual(response.status_code, 404) class MiddlewareExtractClientIpTest(TestCase): def setUp(self): self.middleware = IPRestrictMiddleware() self.factory = RequestFactory() def test_remote_addr_only(self): self.middleware = IPRestrictMiddleware() request = self.factory.get('', REMOTE_ADDR=LOCAL_IP) client_ip = self.middleware.extract_client_ip(request) self.assertEquals(client_ip, LOCAL_IP) @override_settings(TRUSTED_PROXIES=(PROXY,)) def test_single_proxy(self): self.middleware = IPRestrictMiddleware() request = self.factory.get('', REMOTE_ADDR=PROXY, HTTP_X_FORWARDED_FOR = LOCAL_IP) client_ip = self.middleware.extract_client_ip(request) self.assertEquals(client_ip, LOCAL_IP) @override_settings(TRUSTED_PROXIES=(PROXY,'2.2.2.2','4.4.4.4')) def test_multiple_proxies_one_not_trusted(self): self.middleware = IPRestrictMiddleware() proxies = ['2.2.2.2', '3.3.3.3', '4.4.4.4'] request = self.factory.get('', REMOTE_ADDR=PROXY, HTTP_X_FORWARDED_FOR = ', '.join([LOCAL_IP] + proxies)) try: client_ip = self.middleware.extract_client_ip(request) except exceptions.PermissionDenied: pass else: self.fail('Should raise PermissionDenied exception') @override_settings(TRUSTED_PROXIES=(PROXY,'2.2.2.2','3.3.3.3', '4.4.4.4')) def test_multiple_proxies_all_trusted(self): self.middleware = IPRestrictMiddleware() proxies = ['2.2.2.2', '3.3.3.3', '4.4.4.4'] request = self.factory.get('', REMOTE_ADDR=PROXY, HTTP_X_FORWARDED_FOR = ', '.join([LOCAL_IP] + proxies)) client_ip = self.middleware.extract_client_ip(request) self.assertEquals(client_ip, LOCAL_IP) PKH1HO4iprestrict/tests/test_models.pyfrom django.test import TestCase from django.http import HttpResponseForbidden import iprestrict from iprestrict import models class RuleTest(TestCase): def test_restriction_methods_for_allow_rule(self): rule = models.Rule(url_pattern='', action='A') self.assertTrue(rule.is_allowed()) self.assertFalse(rule.is_restricted()) def test_restiction_methods_for_deny_rule(self): rule = models.Rule(url_pattern='', action='D') self.assertFalse(rule.is_allowed()) self.assertTrue(rule.is_restricted()) def test_matches_url_pattern_regex(self): rule = models.Rule(url_pattern='^/pre/[a-d]+[/]?$') self.assertTrue(rule.matches_url('/pre/a/')) self.assertTrue(rule.matches_url('/pre/a')) self.assertFalse(rule.matches_url('/pre/e/')) self.assertFalse(rule.matches_url('/pre/a//')) def test_rank_is_automatically_assigned_on_creation(self): rule1 = models.Rule.objects.create(url_pattern='1', action='A') rule2 = models.Rule.objects.create(url_pattern='2', action='A') rule10 = models.Rule.objects.create(url_pattern='10', action='A', rank=10) rule11 = models.Rule.objects.create(url_pattern='10', action='A') self.assertEquals(rule1.rank, 1) self.assertEquals(rule2.rank, 2) self.assertEquals(rule10.rank, 10) self.assertEquals(rule11.rank, 11) def test_rank_is_not_changed_on_update(self): rule1 = models.Rule.objects.create(url_pattern='1', action='A') rule2 = models.Rule.objects.create(url_pattern='2', action='A') rule1.action = 'D' rule1.save() self.assertEquals(rule1.rank, 1) class RuleWithSampleRulesTests(TestCase): def setUp(self): self.rule1 = models.Rule.objects.create(url_pattern='1', action='A') self.rule2 = models.Rule.objects.create(url_pattern='2', action='A') def test_default_order_is_by_rank(self): rules = models.Rule.objects.all() self.assertEqual(rules[0].url_pattern, '1') self.assertEqual(rules[1].url_pattern, '2') def test_move_up(self): self.rule2.move_up() rules = models.Rule.objects.all() self.assertEqual(rules[0].url_pattern, '2') self.assertEqual(rules[1].url_pattern, '1') def test_move_up_first_rule_does_nothing(self): self.rule1.move_up() rules = models.Rule.objects.all() self.assertEqual(rules[0].url_pattern, '1') self.assertEqual(rules[1].url_pattern, '2') def test_move_down(self): self.rule1.move_down() rules = models.Rule.objects.all() self.assertEqual(rules[0].url_pattern, '2') self.assertEqual(rules[1].url_pattern, '1') def test_move_down_below_default_rule(self): self.rule1.move_down() self.rule1.move_down() rules = models.Rule.objects.all() self.assertEqual(rules[0].url_pattern, '2') self.assertEqual(rules[1].url_pattern, 'ALL') self.assertEqual(rules[2].url_pattern, '1') class IPGroupTest(TestCase): def test_first_ip_group_is_all(self): '''An IP definition matching all should be inserted by default''' all_group = models.IPGroup.objects.get(name='ALL') self.assertTrue(all_group.matches('192.168.1.1')) self.assertTrue(all_group.matches('200.200.200.200')) self.assertTrue(all_group.matches('1.2.3.4')) def test_matches_with_ranges(self): ipgroup = models.IPGroup.objects.create(name='Local IPs') iprange = models.IPRange.objects.create(ip_group=ipgroup, first_ip='192.168.1.1', last_ip='192.168.1.10') iprange2 = models.IPRange.objects.create(ip_group=ipgroup, first_ip='192.168.1.100', last_ip='192.168.1.110') ipgroup.load_ranges() self.assertTrue(ipgroup.matches('192.168.1.1')) self.assertTrue(ipgroup.matches('192.168.1.5')) self.assertTrue(ipgroup.matches('192.168.1.10')) self.assertTrue(ipgroup.matches('192.168.1.105')) self.assertTrue(ipgroup.matches('192.168.1.100')) self.assertFalse(ipgroup.matches('192.168.1.0')) self.assertFalse(ipgroup.matches('192.168.1.11')) self.assertFalse(ipgroup.matches('192.168.1.99')) def test_ipv6_and_ip4_are_separated(self): ipgroup = models.IPGroup.objects.create(name='Local IPs') iprange = models.IPRange.objects.create(ip_group=ipgroup, first_ip='::1') iprange = models.IPRange.objects.create(ip_group=ipgroup, first_ip='0.0.0.2') ipgroup.load_ranges() self.assertFalse(ipgroup.matches('0.0.0.1')) self.assertTrue(ipgroup.matches('0.0.0.2')) # TODO enable when IPv6 conversion to number works #self.assertFalse(ipgroup.matches('::2')) #self.assertTrue(ipgroup.matches('::1')) def test_matches_with_subnets(self): ipgroup = models.IPGroup.objects.create(name='Local IPs') iprange = models.IPRange.objects.create(ip_group=ipgroup, first_ip='192.168.1.0', cidr_prefix_length=30) ipgroup.load_ranges() self.assertTrue(ipgroup.matches('192.168.1.0')) self.assertTrue(ipgroup.matches('192.168.1.1')) self.assertTrue(ipgroup.matches('192.168.1.2')) self.assertTrue(ipgroup.matches('192.168.1.3')) self.assertFalse(ipgroup.matches('192.168.0.255')) self.assertFalse(ipgroup.matches('192.168.1.4')) def test_matches_subnet_first_ip_not_correct(self): ipgroup = models.IPGroup.objects.create(name='Local IPs') iprange = models.IPRange.objects.create(ip_group=ipgroup, first_ip='192.168.1.2', # Should be '192.168.1.1' cidr_prefix_length=30) ipgroup.load_ranges() self.assertTrue(ipgroup.matches('192.168.1.0')) self.assertTrue(ipgroup.matches('192.168.1.1')) self.assertTrue(ipgroup.matches('192.168.1.2')) self.assertTrue(ipgroup.matches('192.168.1.3')) self.assertFalse(ipgroup.matches('192.168.0.255')) self.assertFalse(ipgroup.matches('192.168.1.4')) PKH1HˍQ #iprestrict/tests/test_restrictor.pyfrom django.test import TestCase from django.utils import unittest import iprestrict from iprestrict import models IP = '10.1.1.1' SOME_URL = '/some/url' class IPRestrictorDefaultRulesTest(TestCase): def test_restrictor_restricts_all_by_default(self): restr = iprestrict.IPRestrictor() self.assertTrue(restr.is_restricted('', IP)) self.assertTrue(restr.is_restricted(SOME_URL, IP)) def test_allow_all_ips_for_one_url(self): models.Rule.objects.create(url_pattern=SOME_URL, action='A', rank=1) restr = iprestrict.IPRestrictor() self.assertFalse(restr.is_restricted(SOME_URL, IP)) self.assertFalse(restr.is_restricted(SOME_URL, '192.168.1.1')) def test_caches_rules(self): restr = iprestrict.IPRestrictor() models.Rule.objects.create(url_pattern=SOME_URL, action='A', rank=1) # Restrictor shouldn't read the rules dynamically self.assertTrue(restr.is_restricted(SOME_URL, IP)) def test_reload_rules(self): restr = iprestrict.IPRestrictor() models.Rule.objects.create(url_pattern=SOME_URL, action='A', rank=1) restr.reload_rules() self.assertFalse(restr.is_restricted(SOME_URL, IP)) class IPRestrictorNoRulesTest(TestCase): def setUp(self): models.Rule.objects.all().delete() self.restrictor = iprestrict.IPRestrictor() def test_restrictor_does_not_do_anything_with_empty_rules_table(self): self.assertFalse(self.restrictor.is_restricted('', IP)) self.assertFalse(self.restrictor.is_restricted(SOME_URL, '192.168.1.1')) def test_restrictor_considers_rules_by_rank(self): rule = models.Rule.objects.create(url_pattern='ALL', action='A', rank=2) rule = models.Rule.objects.create(url_pattern='/admin[/].*', action='D', rank=1) restr = iprestrict.IPRestrictor() self.assertFalse(restr.is_restricted('/some/url', IP)) self.assertTrue(restr.is_restricted('/admin/', IP)) self.assertTrue(restr.is_restricted('/admin/somepage', IP)) class IPRestrictorOneUrlAllowedFromOneIpTest(TestCase): def setUp(self): localip = models.IPGroup.objects.create(name='localip') models.IPRange.objects.create(ip_group=localip, first_ip=IP) models.Rule.objects.create(url_pattern=SOME_URL, ip_group=localip, action='A') self.restrictor = iprestrict.IPRestrictor() def test_ip_matches_url_does_not(self): self.assertTrue(self.restrictor.is_restricted('', IP)) def test_url_matches_ip_does_not(self): self.assertTrue(self.restrictor.is_restricted(SOME_URL, '192.168.1.1')) def test_both_match(self): self.assertFalse(self.restrictor.is_restricted(SOME_URL, IP)) def test_reload_if_ipgroup_changed(self): self.assertTrue(self.restrictor.is_restricted(SOME_URL, '10.10.10.10')) localhost = models.IPGroup.objects.get(name='localhost') models.IPRange.objects.create(ip_group=localhost, first_ip='10.10.10.10') self.assertTrue(self.restrictor.is_restricted(SOME_URL, '10.10.10.10')) self.restrictor.reload_rules() self.assertFalse(self.restrictor.is_restricted(SOME_URL, '10.10.10.10')) PKH1H y iprestrict/tests/test_views.pyfrom django.test import TestCase from django.utils.unittest import skip from django.contrib.auth.models import User from iprestrict import models from iprestrict import restrictor class ReloadViewTest(TestCase): IP = '192.168.1.1' def setUp(self): admin = User.objects.create_user('admin', 'admin@nohost.org', 'pass') admin.is_staff = True admin.is_superuser = True admin.save() models.ReloadRulesRequest.request_reload() def add_allow_rule(self): localip = models.IPGroup.objects.create(name='Local IP') models.IPRange.objects.create(ip_group=localip, first_ip=self.IP) models.Rule.objects.create(url_pattern='ALL', ip_group = localip, action='A') def test_reload_view(self): # This test doesn't conform to unit test best practices. # Unfortunately, it does depend on order so it is in one method # instead of 3 methods, each asserting one thing # 1 response = self.client.get('', REMOTE_ADDR = self.IP) self.assertEqual(response.status_code, 403, 'Should be restricted') # 2 self.add_allow_rule() response = self.client.get('', REMOTE_ADDR = self.IP) self.assertEqual(response.status_code, 403, 'Should still be restricted - rules have not been reloaded') # 3 reload rules # TODO I have to log in as superuser for this to work self.client.login(username='admin', password='pass') response = self.client.get('/iprestrict/reload_rules') response = self.client.get('', REMOTE_ADDR = self.IP) self.assertEqual(response.status_code, 404, 'Should be allowed now') PKP1H'ܞhh1django_iprestrict-0.3.2.dist-info/DESCRIPTION.rstDjango app + middleware to restrict access to all or sections of a Django project by client IP ranges PKP1H}/django_iprestrict-0.3.2.dist-info/metadata.json{"classifiers": ["Framework :: Django", "Intended Audience :: Developers", "Intended Audience :: System Administrators", "License :: OSI Approved :: MIT License", "Operating System :: OS Independent", "Topic :: Software Development"], "download_url": "https://github.com/muccg/django-iprestrict/releases", "extensions": {"python.details": {"contacts": [{"email": "devops@ccg.murdoch.edu.au", "name": "Tamas Szabo, CCG, Murdoch University", "role": "author"}], "document_names": {"description": "DESCRIPTION.rst"}, "project_urls": {"Home": "https://github.com/muccg/django-iprestrict"}}}, "extras": [], "generator": "bdist_wheel (0.26.0)", "metadata_version": "2.0", "name": "django-iprestrict", "run_requires": [{"requires": ["django-templatetag-handlebars (==1.2.0)"]}], "summary": "Django app + middleware to restrict access to all or sections of a Django project by client IP ranges", "version": "0.3.2"}PKP1Hw /django_iprestrict-0.3.2.dist-info/top_level.txtiprestrict PKP1H''\\'django_iprestrict-0.3.2.dist-info/WHEELWheel-Version: 1.0 Generator: bdist_wheel (0.26.0) Root-Is-Purelib: true Tag: py2-none-any PKP1H0׎KK*django_iprestrict-0.3.2.dist-info/METADATAMetadata-Version: 2.0 Name: django-iprestrict Version: 0.3.2 Summary: Django app + middleware to restrict access to all or sections of a Django project by client IP ranges Home-page: https://github.com/muccg/django-iprestrict Author: Tamas Szabo, CCG, Murdoch University Author-email: devops@ccg.murdoch.edu.au License: UNKNOWN Download-URL: https://github.com/muccg/django-iprestrict/releases Platform: UNKNOWN Classifier: Framework :: Django Classifier: Intended Audience :: Developers Classifier: Intended Audience :: System Administrators Classifier: License :: OSI Approved :: MIT License Classifier: Operating System :: OS Independent Classifier: Topic :: Software Development Requires-Dist: django-templatetag-handlebars (==1.2.0) Django app + middleware to restrict access to all or sections of a Django project by client IP ranges PKP1H**(django_iprestrict-0.3.2.dist-info/RECORDdjango_iprestrict-0.3.2.dist-info/DESCRIPTION.rst,sha256=MW1Mpxmn0Ho1mDJ2PJjKS20MmZ8tb4X3lAVDelDxU-Y,104 django_iprestrict-0.3.2.dist-info/METADATA,sha256=boFYMytp9Meee5fhmzUY1-aRIJzIshivFO4lyGtUOWY,843 django_iprestrict-0.3.2.dist-info/RECORD,, django_iprestrict-0.3.2.dist-info/WHEEL,sha256=JTb7YztR8fkPg6aSjc571Q4eiVHCwmUDlX8PhuuqIIE,92 django_iprestrict-0.3.2.dist-info/metadata.json,sha256=vXC2oo6n9LDhzX6xEUtD0yULH2ArqjqxpbbiPm3P0d8,907 django_iprestrict-0.3.2.dist-info/top_level.txt,sha256=CeJEtZeoqyJUxGxI__2rkygfrHfNhrmZITWotbsleNA,11 iprestrict/__init__.py,sha256=owF4OsoHNSz1-uLzEdKlf74dOdUAWmIJcIG258Tax_w,37 iprestrict/admin.py,sha256=mmITWz3s1EQu_ZQBja8-Ij0dZzU8-bLvqXvZ_MORP9w,2388 iprestrict/decorators.py,sha256=tOXsPaFcN69QLBPlruQI0s0OZW77D8mZVnq9wDEYvAk,1514 iprestrict/ip_utils.py,sha256=ZgXP6HDNOyGwyv76a9J2rdmw6wYTenb_2f75jVstivo,739 iprestrict/middleware.py,sha256=OG6J4gbEUXOZrn0HrZB248RAtwofAW7-NKU9RTHJ2Yc,1825 iprestrict/models.py,sha256=MGY18wrCMA-Js1l8F4XdJGbbNGhFghCx2HS8sEbE-B4,5443 iprestrict/restrictor.py,sha256=edMvSFcFGFtZ7wT2a4_oG4hcJZQEAXkOQUQRsJlQVi0,568 iprestrict/test_settings.py,sha256=53LFkSXGgQ6AvJ23Pc2Cp4iHdyAoEcmLCpUeqolc7oE,5929 iprestrict/test_settings_1_3_1.py,sha256=feb774zzjlUD0NLB9Ir7bc80EhHotYEsB2QdGpejcG0,5125 iprestrict/test_urls.py,sha256=YInktE40unB1HEjTujhomT6v_Ko2DuCHQMViOQQT_WE,702 iprestrict/urls.py,sha256=FVOZSu8K-9Hf0QEaEDeLo3yIBu8pS34YcCtpqdcWVsA,361 iprestrict/views.py,sha256=mvHBEJ2zkbRbi7j2VJSLhy1Bft1RHKO6Ufo6mn0NNc8,2323 iprestrict/fixtures/default_data.json,sha256=3UA5Qx3eDZ208mkB8vNZiiw-VwHpo9bkBDOh2bZKe2A,1600 iprestrict/management/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 iprestrict/management/commands/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 iprestrict/management/commands/importrules.py,sha256=fGmMkGXBQqkEQsdtAdd3BFdPAQk1UwmPr6cW1fBDddQ,807 iprestrict/management/commands/reloadrules.py,sha256=lU71zPk-xB9rc57oM6xUgUaYC6HjQjpk3EKEIWmE9cQ,442 iprestrict/migrations/0001_initial.py,sha256=0Xc2ajroRPfe0M0zKWEi17nEjtORqtyov8sbwQK5aMs,4653 iprestrict/migrations/0002_create_initial_data.py,sha256=ji5-4M7lq1DF5TervLo0YJJ6SfTRQrtdNrRdbKz0xp4,3397 iprestrict/migrations/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 iprestrict/south_migrations/0001_initial.py,sha256=0Xc2ajroRPfe0M0zKWEi17nEjtORqtyov8sbwQK5aMs,4653 iprestrict/south_migrations/0002_create_initial_data.py,sha256=ji5-4M7lq1DF5TervLo0YJJ6SfTRQrtdNrRdbKz0xp4,3397 iprestrict/south_migrations/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 iprestrict/static/css/test_rules.css,sha256=X46J3SDoEhioIa41nItDTQvV-yygGrlq-dfh5ygfgWc,885 iprestrict/static/javascript/lib/handlebars.js,sha256=UMNLYUU--vPODHfsOtoQC2n70xWtEmS3J7B8X8t_eRI,49699 iprestrict/static/javascript/lib/jquery-min.js,sha256=1z4uG_-cVbhShP8ofLINwprZFl7AkJGgWXthGZ8zCAU,92556 iprestrict/templates/404.html,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 iprestrict/templates/iprestrict/test_rules.html,sha256=q4YaZ0ACo_Zm2ErW1EFISNaQW_pdHTO528p58--Sryw,2748 iprestrict/tests/__init__.py,sha256=AbpHGcgLb-kRsJGnwFEktk7uzpZOCcBY74-YBdrKVGs,1 iprestrict/tests/test_ip_utils.py,sha256=ZAtzSDDCSEoAojFzl4CJgcFpRnbfpxPzz3EozqAnxf8,952 iprestrict/tests/test_middleware.py,sha256=VgP26qQih4ATxZkLdN4Mn7TBoMyrZQsHBXypRw8Dfok,5080 iprestrict/tests/test_models.py,sha256=P4LeUBRbdoYW1qJ9TnjNzz52PLNK-CtBoCEjQ56N8dQ,6109 iprestrict/tests/test_restrictor.py,sha256=lJUCfa71g1xf_XnLN0V4Y208RAX6Ie-iP6ytn-tA8hE,3242 iprestrict/tests/test_views.py,sha256=pjRto065qcx_DmDYO63QLqHKI8WN94l9d8SRUtuuubk,1676 PKt[CR;%%iprestrict/__init__.pyPKH1H>T T Yiprestrict/admin.pyPKH1HWw iprestrict/decorators.pyPKH1HB6iprestrict/ip_utils.pyPKH1HB!!iprestrict/middleware.pyPKH1HܗCCliprestrict/models.pyPKH1Hl88/iprestrict/restrictor.pyPKH1H))O2iprestrict/test_settings.pyPKH1HU!Iiprestrict/test_settings_1_3_1.pyPKH1H" ]iprestrict/test_urls.pyPKH1Hii`iprestrict/urls.pyPKH1H3  biprestrict/views.pyPKt[C!kiprestrict/management/__init__.pyPKt[C*liprestrict/management/commands/__init__.pyPKt[CE[''-Lliprestrict/management/commands/importrules.pyPKt[C$-oiprestrict/management/commands/reloadrules.pyPKt[Cqiprestrict/templates/404.htmlPKt[CЄ /qiprestrict/templates/iprestrict/test_rules.htmlPKt[Ccuu$}iprestrict/static/css/test_rules.cssPKt[C1##.iprestrict/static/javascript/lib/handlebars.jsPKt[Co-ii.-Ciprestrict/static/javascript/lib/jquery-min.jsPKH1H%@@%iprestrict/fixtures/default_data.jsonPKt[C'r--%iprestrict/migrations/0001_initial.pyPKt[C㨺E E 1iprestrict/migrations/0002_create_initial_data.pyPKt[C!iprestrict/migrations/__init__.pyPKH1H'r--+iprestrict/south_migrations/0001_initial.pyPKH1H㨺E E 7Aiprestrict/south_migrations/0002_create_initial_data.pyPKH1H'iprestrict/south_migrations/__init__.pyPKH1H2 iprestrict/tests/__init__.pyPKH1H{k![iprestrict/tests/test_ip_utils.pyPKH1H X!s#Riprestrict/tests/test_middleware.pyPKH1HO4k iprestrict/tests/test_models.pyPKH1HˍQ #$iprestrict/tests/test_restrictor.pyPKH1H y p1iprestrict/tests/test_views.pyPKP1H'ܞhh188django_iprestrict-0.3.2.dist-info/DESCRIPTION.rstPKP1H}/8django_iprestrict-0.3.2.dist-info/metadata.jsonPKP1Hw /<django_iprestrict-0.3.2.dist-info/top_level.txtPKP1H''\\'=django_iprestrict-0.3.2.dist-info/WHEELPKP1H0׎KK*=django_iprestrict-0.3.2.dist-info/METADATAPKP1H**(SAdjango_iprestrict-0.3.2.dist-info/RECORDPK(( O