[{"data":1,"prerenderedAt":12050},["ShallowReactive",2],{"navigation":3,"/admin-site/admin-security-best-practices":1016,"/admin-site/admin-security-best-practices-surround":12045},[4,54,89,123,165,211,253,331,389,439,462,488,514,540,578,624,650,672,702,736,774,800,850,856,862,868,874,928],{"title":5,"path":6,"stem":7,"children":8},"Introduction and Foundations","/introduction-and-foundations","01.introduction-and-foundations/01.index",[9,10,14,18,22,26,30,34,38,42,46,50],{"title":5,"path":6,"stem":7},{"title":11,"path":12,"stem":13},"What is Django","/introduction-and-foundations/what-is-django","01.introduction-and-foundations/02.what-is-django",{"title":15,"path":16,"stem":17},"Key Concepts and Philosophy","/introduction-and-foundations/key-concepts-and-philosophy","01.introduction-and-foundations/03.key-concepts-and-philosophy",{"title":19,"path":20,"stem":21},"MVC vs MVT: Understanding Django's Architecture","/introduction-and-foundations/mvc-vs-mvt","01.introduction-and-foundations/04.mvc-vs-mvt",{"title":23,"path":24,"stem":25},"Project Structure Overview","/introduction-and-foundations/project-structure-overview","01.introduction-and-foundations/05.project-structure-overview",{"title":27,"path":28,"stem":29},"Installing Django","/introduction-and-foundations/installing-django","01.introduction-and-foundations/06.installing-django",{"title":31,"path":32,"stem":33},"Creating Your First Django Project","/introduction-and-foundations/creating-first-project","01.introduction-and-foundations/07.creating-first-project",{"title":35,"path":36,"stem":37},"Creating Your First Django App","/introduction-and-foundations/creating-first-app","01.introduction-and-foundations/08.creating-first-app",{"title":39,"path":40,"stem":41},"Your First Django \"Hello World\"","/introduction-and-foundations/hello-world","01.introduction-and-foundations/09.hello-world",{"title":43,"path":44,"stem":45},"Django Quick Start Guide","/introduction-and-foundations/quick-start-guide","01.introduction-and-foundations/10.quick-start-guide",{"title":47,"path":48,"stem":49},"How Django Handles Requests","/introduction-and-foundations/how-django-handles-requests","01.introduction-and-foundations/11.how-django-handles-requests",{"title":51,"path":52,"stem":53},"Django Settings","/introduction-and-foundations/django-settings","01.introduction-and-foundations/12.django-settings",{"title":55,"path":56,"stem":57,"children":58},"The Development Environment","/development-environment","02.development-environment/1.index",[59,61,65,69,73,77,81,85],{"title":60,"path":56,"stem":57},"Development Environment",{"title":62,"path":63,"stem":64},"Recommended Tooling for Django Development","/development-environment/recommended-tooling","02.development-environment/2.recommended-tooling",{"title":66,"path":67,"stem":68},"Virtual Environments","/development-environment/virtual-environments","02.development-environment/3.virtual-environments",{"title":70,"path":71,"stem":72},"Django Admin and Management Commands","/development-environment/admin-and-management-commands","02.development-environment/4.admin-and-management-commands",{"title":74,"path":75,"stem":76},"Django Project Settings","/development-environment/project-settings","02.development-environment/5.project-settings",{"title":78,"path":79,"stem":80},"Managing Django Environments: Local, Staging, and Production","/development-environment/environments-local-staging-production","02.development-environment/6.environments-local-staging-production",{"title":82,"path":83,"stem":84},"Running Django Development Server","/development-environment/running-development-server","02.development-environment/7.running-development-server",{"title":86,"path":87,"stem":88},"Working with Django Shell","/development-environment/working-with-shell","02.development-environment/8.working-with-shell",{"title":90,"path":91,"stem":92,"children":93},"Templates and Presentation Layer","/templates-and-presentation","03.templates-and-presentation/1.index",[94,95,99,103,107,111,115,119],{"title":90,"path":91,"stem":92},{"title":96,"path":97,"stem":98},"Introduction to Django Templates","/templates-and-presentation/introduction-to-templates","03.templates-and-presentation/2.introduction-to-templates",{"title":100,"path":101,"stem":102},"The Django Template Language","/templates-and-presentation/django-template-language","03.templates-and-presentation/3.django-template-language",{"title":104,"path":105,"stem":106},"Template Inheritance","/templates-and-presentation/template-inheritance","03.templates-and-presentation/4.template-inheritance",{"title":108,"path":109,"stem":110},"Built-in Template Tags and Filters","/templates-and-presentation/built-in-template-tags-and-filters","03.templates-and-presentation/5.built-in-template-tags-and-filters",{"title":112,"path":113,"stem":114},"Including Static Files","/templates-and-presentation/including-static-files","03.templates-and-presentation/6.including-static-files",{"title":116,"path":117,"stem":118},"Working with Media Files","/templates-and-presentation/working-with-media-files","03.templates-and-presentation/7.working-with-media-files",{"title":120,"path":121,"stem":122},"Using Alternative Template Engines","/templates-and-presentation/using-alternative-template-engines","03.templates-and-presentation/8.using-alternative-template-engines",{"title":124,"path":125,"stem":126,"children":127},"URLs and Views","/urls-and-views","04.urls-and-views/01.index",[128,129,133,137,141,145,149,153,157,161],{"title":124,"path":125,"stem":126},{"title":130,"path":131,"stem":132},"The URL Dispatcher","/urls-and-views/the-url-dispatcher","04.urls-and-views/02.the-url-dispatcher",{"title":134,"path":135,"stem":136},"Writing Function-Based Views","/urls-and-views/writing-function-based-views","04.urls-and-views/03.writing-function-based-views",{"title":138,"path":139,"stem":140},"View Decorators","/urls-and-views/view-decorators","04.urls-and-views/04.view-decorators",{"title":142,"path":143,"stem":144},"Rendering Responses","/urls-and-views/rendering-responses","04.urls-and-views/05.rendering-responses",{"title":146,"path":147,"stem":148},"Redirects","/urls-and-views/redirects","04.urls-and-views/06.redirects",{"title":150,"path":151,"stem":152},"Handling HTTP Methods","/urls-and-views/handling-http-methods","04.urls-and-views/07.handling-http-methods",{"title":154,"path":155,"stem":156},"Conditional View Processing","/urls-and-views/conditional-view-processing","04.urls-and-views/08.conditional-view-processing",{"title":158,"path":159,"stem":160},"File Uploads","/urls-and-views/file-uploads","04.urls-and-views/09.file-uploads",{"title":162,"path":163,"stem":164},"Using Django Shortcut Functions","/urls-and-views/using-django-shortcut-functions","04.urls-and-views/10.using-django-shortcut-functions",{"title":166,"path":167,"stem":168,"children":169},"Class Based Views","/class-based-views","05.class-based-views/01.index",[170,171,175,179,183,187,191,195,199,203,207],{"title":166,"path":167,"stem":168},{"title":172,"path":173,"stem":174},"Introduction to Class-Based Views","/class-based-views/introduction-to-class-based-views","05.class-based-views/02.introduction-to-class-based-views",{"title":176,"path":177,"stem":178},"Common Base Classes","/class-based-views/common-base-classes","05.class-based-views/03.common-base-classes",{"title":180,"path":181,"stem":182},"Built-in Generic Views","/class-based-views/built-in-generic-views","05.class-based-views/04.built-in-generic-views",{"title":184,"path":185,"stem":186},"Views for CRUD Operations","/class-based-views/views-for-crud-operations","05.class-based-views/05.views-for-crud-operations",{"title":188,"path":189,"stem":190},"Handling Forms with Class-Based Views","/class-based-views/handling-forms-with-class-based-views","05.class-based-views/06.handling-forms-with-class-based-views",{"title":192,"path":193,"stem":194},"Using Mixins","/class-based-views/using-mixins","05.class-based-views/07.using-mixins",{"title":196,"path":197,"stem":198},"URL Configuration with Class-Based Views","/class-based-views/url-configuration-with-class-based-views","05.class-based-views/08.url-configuration-with-class-based-views",{"title":200,"path":201,"stem":202},"Subclassing Generic Views","/class-based-views/subclassing-generic-views","05.class-based-views/09.subclassing-generic-views",{"title":204,"path":205,"stem":206},"Asynchronous Class-Based Views","/class-based-views/asynchronous-class-based-views","05.class-based-views/10.asynchronous-class-based-views",{"title":208,"path":209,"stem":210},"Pagination","/class-based-views/pagination","05.class-based-views/11.pagination",{"title":212,"path":213,"stem":214,"children":215},"Forms and User Input","/forms-and-user-input","06.forms-and-user-input/01.index",[216,217,221,225,229,233,237,241,245,249],{"title":212,"path":213,"stem":214},{"title":218,"path":219,"stem":220},"Understanding HTML Forms","/forms-and-user-input/understanding-html-forms","06.forms-and-user-input/02.understanding-html-forms",{"title":222,"path":223,"stem":224},"Django's Role in Form Handling","/forms-and-user-input/djangos-role-in-form-handling","06.forms-and-user-input/03.djangos-role-in-form-handling",{"title":226,"path":227,"stem":228},"Creating Forms with Forms API","/forms-and-user-input/creating-forms-with-forms-api","06.forms-and-user-input/04.creating-forms-with-forms-api",{"title":230,"path":231,"stem":232},"Form Validation","/forms-and-user-input/form-validation","06.forms-and-user-input/05.form-validation",{"title":234,"path":235,"stem":236},"Built-in Fields and Widgets","/forms-and-user-input/built-in-fields-and-widgets","06.forms-and-user-input/06.built-in-fields-and-widgets",{"title":238,"path":239,"stem":240},"Form Rendering in Templates","/forms-and-user-input/form-rendering-in-templates","06.forms-and-user-input/07.form-rendering-in-templates",{"title":242,"path":243,"stem":244},"Model Forms","/forms-and-user-input/model-forms","06.forms-and-user-input/08.model-forms",{"title":246,"path":247,"stem":248},"Advanced Form Techniques","/forms-and-user-input/advanced-form-techniques","06.forms-and-user-input/09.advanced-form-techniques",{"title":250,"path":251,"stem":252},"Security Considerations for Forms","/forms-and-user-input/security-considerations-for-forms","06.forms-and-user-input/10.security-considerations-for-forms",{"title":254,"path":255,"stem":256,"children":257},"Models and Databases","/models-and-databases","07.models-and-databases/01.index",[258,259,263,267,271,275,279,283,287,291,295,299,303,307,311,315,319,323,327],{"title":254,"path":255,"stem":256},{"title":260,"path":261,"stem":262},"Understanding Django Models","/models-and-databases/understanding-django-models","07.models-and-databases/02.understanding-django-models",{"title":264,"path":265,"stem":266},"Defining Fields","/models-and-databases/defining-fields","07.models-and-databases/03.defining-fields",{"title":268,"path":269,"stem":270},"Relationships and Foreign Keys","/models-and-databases/relationships-and-foreign-keys","07.models-and-databases/04.relationships-and-foreign-keys",{"title":272,"path":273,"stem":274},"Examples of Relationship Patterns","/models-and-databases/examples-of-relationship-patterns","07.models-and-databases/05.examples-of-relationship-patterns",{"title":276,"path":277,"stem":278},"Making Queries","/models-and-databases/making-queries","07.models-and-databases/06.making-queries",{"title":280,"path":281,"stem":282},"Filtering, Ordering, and Slicing","/models-and-databases/filtering-ordering-slicing","07.models-and-databases/07.filtering-ordering-slicing",{"title":284,"path":285,"stem":286},"Managers and QuerySets","/models-and-databases/managers-and-querysets","07.models-and-databases/08.managers-and-querysets",{"title":288,"path":289,"stem":290},"Aggregation","/models-and-databases/aggregation","07.models-and-databases/09.aggregation",{"title":292,"path":293,"stem":294},"Search","/models-and-databases/search","07.models-and-databases/10.search",{"title":296,"path":297,"stem":298},"Raw SQL Queries","/models-and-databases/raw-sql-queries","07.models-and-databases/11.raw-sql-queries",{"title":300,"path":301,"stem":302},"Transactions","/models-and-databases/transactions","07.models-and-databases/12.transactions",{"title":304,"path":305,"stem":306},"Multiple Databases","/models-and-databases/multiple-databases","07.models-and-databases/13.multiple-databases",{"title":308,"path":309,"stem":310},"Tablespaces","/models-and-databases/tablespaces","07.models-and-databases/14.tablespaces",{"title":312,"path":313,"stem":314},"Composite Primary Keys","/models-and-databases/composite-primary-keys","07.models-and-databases/15.composite-primary-keys",{"title":316,"path":317,"stem":318},"Database Instrumentation","/models-and-databases/database-instrumentation","07.models-and-databases/16.database-instrumentation",{"title":320,"path":321,"stem":322},"Database Optimization","/models-and-databases/database-optimization","07.models-and-databases/17.database-optimization",{"title":324,"path":325,"stem":326},"Fixtures","/models-and-databases/fixtures","07.models-and-databases/18.fixtures",{"title":328,"path":329,"stem":330},"Signals","/models-and-databases/signals","07.models-and-databases/19.signals",{"title":332,"path":333,"stem":334,"children":335},"Migrations","/migrations","08.migrations/01.index",[336,337,341,345,349,353,357,361,365,369,373,377,381,385],{"title":332,"path":333,"stem":334},{"title":338,"path":339,"stem":340},"How Migrations Work","/migrations/how-migrations-work","08.migrations/02.how-migrations-work",{"title":342,"path":343,"stem":344},"Management Commands","/migrations/management-commands","08.migrations/03.management-commands",{"title":346,"path":347,"stem":348},"Dependencies and Workflow","/migrations/dependencies-and-workflow","08.migrations/04.dependencies-and-workflow",{"title":350,"path":351,"stem":352},"Transaction Handling","/migrations/transaction-handling","08.migrations/05.transaction-handling",{"title":354,"path":355,"stem":356},"Adding Migrations to Apps","/migrations/adding-migrations-to-apps","08.migrations/06.adding-migrations-to-apps",{"title":358,"path":359,"stem":360},"Reversing Migrations","/migrations/reversing-migrations","08.migrations/07.reversing-migrations",{"title":362,"path":363,"stem":364},"Historical Models","/migrations/historical-models","08.migrations/08.historical-models",{"title":366,"path":367,"stem":368},"Considerations When Removing Fields","/migrations/considerations-when-removing-fields","08.migrations/09.considerations-when-removing-fields",{"title":370,"path":371,"stem":372},"Data Migrations","/migrations/data-migrations","08.migrations/10.data-migrations",{"title":374,"path":375,"stem":376},"Squashing Migrations","/migrations/squashing-migrations","08.migrations/11.squashing-migrations",{"title":378,"path":379,"stem":380},"Serializing Values","/migrations/serializing-values","08.migrations/12.serializing-values",{"title":382,"path":383,"stem":384},"Supporting Multiple Django Versions","/migrations/supporting-multiple-django-versions","08.migrations/13.supporting-multiple-django-versions",{"title":386,"path":387,"stem":388},"Django Serialization Framework","/migrations/django-serialization-framework","08.migrations/14.django-serialization-framework",{"title":390,"path":391,"stem":392,"children":393},"Authentication and Authorization","/authentication-and-authorization","09.authentication-and-authorization/01.index",[394,395,399,403,407,411,415,419,423,427,431,435],{"title":390,"path":391,"stem":392},{"title":396,"path":397,"stem":398},"Overview of Django's Authentication System","/authentication-and-authorization/overview-of-django-authentication-system","09.authentication-and-authorization/02.overview-of-django-authentication-system",{"title":400,"path":401,"stem":402},"Users and Groups","/authentication-and-authorization/users-and-groups","09.authentication-and-authorization/03.users-and-groups",{"title":404,"path":405,"stem":406},"Permissions","/authentication-and-authorization/permissions","09.authentication-and-authorization/04.permissions",{"title":408,"path":409,"stem":410},"Password Management","/authentication-and-authorization/password-management","09.authentication-and-authorization/05.password-management",{"title":412,"path":413,"stem":414},"Authentication Views","/authentication-and-authorization/authentication-views","09.authentication-and-authorization/06.authentication-views",{"title":416,"path":417,"stem":418},"Login and Logout","/authentication-and-authorization/login-and-logout","09.authentication-and-authorization/07.login-and-logout",{"title":420,"path":421,"stem":422},"Custom User Models","/authentication-and-authorization/custom-user-models","09.authentication-and-authorization/08.custom-user-models",{"title":424,"path":425,"stem":426},"Middleware for Authentication","/authentication-and-authorization/middleware-for-authentication","09.authentication-and-authorization/09.middleware-for-authentication",{"title":428,"path":429,"stem":430},"Authorization in Views and Templates","/authentication-and-authorization/authorization-in-views-and-templates","09.authentication-and-authorization/10.authorization-in-views-and-templates",{"title":432,"path":433,"stem":434},"Integrating Social Authentication","/authentication-and-authorization/integrating-social-authentication","09.authentication-and-authorization/11.integrating-social-authentication",{"title":436,"path":437,"stem":438},"Security Best Practices","/authentication-and-authorization/security-best-practices","09.authentication-and-authorization/12.security-best-practices",{"title":440,"path":441,"stem":442,"children":443},"Sessions, Cookies, and State","/sessions-cookies-state","10.sessions-cookies-state/01.index",[444,446,450,454,458],{"title":445,"path":441,"stem":442},"Sessions, Cookies, and State Management",{"title":447,"path":448,"stem":449},"Introduction to Sessions","/sessions-cookies-state/introduction-to-sessions","10.sessions-cookies-state/02.introduction-to-sessions",{"title":451,"path":452,"stem":453},"Working with Cookies","/sessions-cookies-state/working-with-cookies","10.sessions-cookies-state/03.working-with-cookies",{"title":455,"path":456,"stem":457},"Server-Side Session Storage Options","/sessions-cookies-state/server-side-session-storage-options","10.sessions-cookies-state/04.server-side-session-storage-options",{"title":459,"path":460,"stem":461},"Session Security","/sessions-cookies-state/session-security","10.sessions-cookies-state/05.session-security",{"title":463,"path":464,"stem":465,"children":466},"Working with Files","/working-with-files","11.working-with-files/01.index",[467,468,472,476,480,484],{"title":463,"path":464,"stem":465},{"title":469,"path":470,"stem":471},"Files in Models","/working-with-files/files-in-models","11.working-with-files/02.files-in-models",{"title":473,"path":474,"stem":475},"The File Object","/working-with-files/the-file-object","11.working-with-files/03.the-file-object",{"title":477,"path":478,"stem":479},"Storage Backends","/working-with-files/storage-backends","11.working-with-files/04.storage-backends",{"title":481,"path":482,"stem":483},"Using Cloud Storage Providers","/working-with-files/using-cloud-storage-providers","11.working-with-files/05.using-cloud-storage-providers",{"title":485,"path":486,"stem":487},"Managing Media in Production","/working-with-files/managing-media-in-production","11.working-with-files/06.managing-media-in-production",{"title":489,"path":490,"stem":491,"children":492},"Admin Site","/admin-site","12.admin-site/01.index",[493,494,498,502,506,510],{"title":489,"path":490,"stem":491},{"title":495,"path":496,"stem":497},"Enabling the Admin","/admin-site/enabling-the-admin","12.admin-site/02.enabling-the-admin",{"title":499,"path":500,"stem":501},"Registering Models","/admin-site/registering-models","12.admin-site/03.registering-models",{"title":503,"path":504,"stem":505},"Customizing Admin Display","/admin-site/customizing-admin-display","12.admin-site/04.customizing-admin-display",{"title":507,"path":508,"stem":509},"Admin Actions","/admin-site/admin-actions","12.admin-site/05.admin-actions",{"title":511,"path":512,"stem":513},"Admin Security Best Practices","/admin-site/admin-security-best-practices","12.admin-site/06.admin-security-best-practices",{"title":515,"path":516,"stem":517,"children":518},"Middleware","/middleware","13.middleware/01.index",[519,520,524,528,532,536],{"title":515,"path":516,"stem":517},{"title":521,"path":522,"stem":523},"Middleware Overview","/middleware/middleware-overview","13.middleware/02.middleware-overview",{"title":525,"path":526,"stem":527},"Built-in Middleware","/middleware/built-in-middleware","13.middleware/03.built-in-middleware",{"title":529,"path":530,"stem":531},"Creating Custom Middleware","/middleware/creating-custom-middleware","13.middleware/04.creating-custom-middleware",{"title":533,"path":534,"stem":535},"Middleware Ordering","/middleware/middleware-ordering","13.middleware/05.middleware-ordering",{"title":537,"path":538,"stem":539},"Performance and Debugging","/middleware/performance-and-debugging","13.middleware/06.performance-and-debugging",{"title":541,"path":542,"stem":543,"children":544},"Security","/security","14.security/01.index",[545,546,550,554,558,562,566,570,574],{"title":541,"path":542,"stem":543},{"title":547,"path":548,"stem":549},"Django Security Philosophy","/security/django-security-philosophy","14.security/02.django-security-philosophy",{"title":551,"path":552,"stem":553},"Cross Site Request Forgery","/security/cross-site-request-forgery","14.security/03.cross-site-request-forgery",{"title":555,"path":556,"stem":557},"Cross Site Scripting","/security/cross-site-scripting","14.security/04.cross-site-scripting",{"title":559,"path":560,"stem":561},"SQL Injection Protection","/security/sql-injection-protection","14.security/05.sql-injection-protection",{"title":563,"path":564,"stem":565},"Clickjacking Protection","/security/clickjacking-protection","14.security/06.clickjacking-protection",{"title":567,"path":568,"stem":569},"HTTPS Setup and HSTS","/security/https-setup-and-hsts","14.security/07.https-setup-and-hsts",{"title":571,"path":572,"stem":573},"Password Storage and Cryptography","/security/password-storage-and-cryptography","14.security/08.password-storage-and-cryptography",{"title":575,"path":576,"stem":577},"Secure Deployment Checklist","/security/secure-deployment-checklist","14.security/09.secure-deployment-checklist",{"title":579,"path":580,"stem":581,"children":582},"Testing","/testing","15.testing/01.index",[583,584,588,592,596,600,604,608,612,616,620],{"title":579,"path":580,"stem":581},{"title":585,"path":586,"stem":587},"Introduction to Django Testing","/testing/introduction-to-django-testing","15.testing/02.introduction-to-django-testing",{"title":589,"path":590,"stem":591},"Writing and Running Tests","/testing/writing-and-running-tests","15.testing/03.writing-and-running-tests",{"title":593,"path":594,"stem":595},"Test Tools","/testing/test-tools","15.testing/04.test-tools",{"title":597,"path":598,"stem":599},"Testing Models","/testing/testing-models","15.testing/05.testing-models",{"title":601,"path":602,"stem":603},"Testing Views","/testing/testing-views","15.testing/06.testing-views",{"title":605,"path":606,"stem":607},"Testing Forms","/testing/testing-forms","15.testing/07.testing-forms",{"title":609,"path":610,"stem":611},"Testing Templates","/testing/testing-templates","15.testing/08.testing-templates",{"title":613,"path":614,"stem":615},"Testing Authentication","/testing/testing-authentication","15.testing/09.testing-authentication",{"title":617,"path":618,"stem":619},"Advanced Testing Topics","/testing/advanced-testing-topics","15.testing/10.advanced-testing-topics",{"title":621,"path":622,"stem":623},"Performance Testing","/testing/performance-testing","15.testing/11.performance-testing",{"title":625,"path":626,"stem":627,"children":628},"Static Assets and Frontend Integration","/static-assets-and-frontend-integration","16.static-assets-and-frontend-integration/01.index",[629,630,634,638,642,646],{"title":625,"path":626,"stem":627},{"title":631,"path":632,"stem":633},"Working with Static Files","/static-assets-and-frontend-integration/working-with-static-files","16.static-assets-and-frontend-integration/02.working-with-static-files",{"title":635,"path":636,"stem":637},"Integrating CSS and JavaScript","/static-assets-and-frontend-integration/integrating-css-and-javascript","16.static-assets-and-frontend-integration/03.integrating-css-and-javascript",{"title":639,"path":640,"stem":641},"Using Build Tools like Vite or Webpack","/static-assets-and-frontend-integration/using-build-tools-like-vite-or-webpack","16.static-assets-and-frontend-integration/04.using-build-tools-like-vite-or-webpack",{"title":643,"path":644,"stem":645},"Using React or Vue with Django","/static-assets-and-frontend-integration/using-react-or-vue-with-django","16.static-assets-and-frontend-integration/05.using-react-or-vue-with-django",{"title":647,"path":648,"stem":649},"Managing CORS","/static-assets-and-frontend-integration/managing-cors","16.static-assets-and-frontend-integration/06.managing-cors",{"title":651,"path":652,"stem":653,"children":654},"Internationalization and Localization","/internationalization-and-localization","17.internationalization-and-localization/01.index",[655,656,660,664,668],{"title":651,"path":652,"stem":653},{"title":657,"path":658,"stem":659},"Enabling Translation","/internationalization-and-localization/enabling-translation","17.internationalization-and-localization/02.enabling-translation",{"title":661,"path":662,"stem":663},"Translating Text in Code and Templates","/internationalization-and-localization/translating-text-in-code-and-templates","17.internationalization-and-localization/03.translating-text-in-code-and-templates",{"title":665,"path":666,"stem":667},"Timezone Support","/internationalization-and-localization/timezone-support","17.internationalization-and-localization/04.timezone-support",{"title":669,"path":670,"stem":671},"Locale Middleware","/internationalization-and-localization/locale-middleware","17.internationalization-and-localization/05.locale-middleware",{"title":673,"path":674,"stem":675,"children":676},"Caching","/caching","18.caching/01.index",[677,678,682,686,690,694,698],{"title":673,"path":674,"stem":675},{"title":679,"path":680,"stem":681},"Introduction to Caching","/caching/introduction-to-caching","18.caching/02.introduction-to-caching",{"title":683,"path":684,"stem":685},"Cache Backends","/caching/cache-backends","18.caching/03.cache-backends",{"title":687,"path":688,"stem":689},"Per View Caching","/caching/per-view-caching","18.caching/04.per-view-caching",{"title":691,"path":692,"stem":693},"Low Level Cache API","/caching/low-level-cache-api","18.caching/05.low-level-cache-api",{"title":695,"path":696,"stem":697},"Template Fragment Caching","/caching/template-fragment-caching","18.caching/06.template-fragment-caching",{"title":699,"path":700,"stem":701},"Deployment Level Caching Patterns","/caching/deployment-level-caching-patterns","18.caching/07.deployment-level-caching-patterns",{"title":703,"path":704,"stem":705,"children":706},"Asynchronous Django","/asynchronous-django","19.asynchronous-django/01.index",[707,708,712,716,720,724,728,732],{"title":703,"path":704,"stem":705},{"title":709,"path":710,"stem":711},"Introduction to ASGI","/asynchronous-django/introduction-to-asgi","19.asynchronous-django/02.introduction-to-asgi",{"title":713,"path":714,"stem":715},"Async Views","/asynchronous-django/async-views","19.asynchronous-django/03.async-views",{"title":717,"path":718,"stem":719},"Async ORM Status","/asynchronous-django/async-orm-status","19.asynchronous-django/04.async-orm-status",{"title":721,"path":722,"stem":723},"WebSockets with Channels","/asynchronous-django/websockets-with-channels","19.asynchronous-django/05.websockets-with-channels",{"title":725,"path":726,"stem":727},"Background Tasks with Celery or RQ","/asynchronous-django/background-tasks-with-celery-or-rq","19.asynchronous-django/06.background-tasks-with-celery-or-rq",{"title":729,"path":730,"stem":731},"Asynchronous Support","/asynchronous-django/asynchronous-support","19.asynchronous-django/07.asynchronous-support",{"title":733,"path":734,"stem":735},"Django's Tasks Framework","/asynchronous-django/django-tasks-framework","19.asynchronous-django/08.django-tasks-framework",{"title":737,"path":738,"stem":739,"children":740},"Deployment","/deployment","20.deployment/01.index",[741,742,746,750,754,758,762,766,770],{"title":737,"path":738,"stem":739},{"title":743,"path":744,"stem":745},"Preparing for Production","/deployment/preparing-for-production","20.deployment/02.preparing-for-production",{"title":747,"path":748,"stem":749},"Using WSGI and ASGI Servers","/deployment/using-wsgi-and-asgi-servers","20.deployment/03.using-wsgi-and-asgi-servers",{"title":751,"path":752,"stem":753},"Deploying on Linux Servers","/deployment/deploying-on-linux-servers","20.deployment/04.deploying-on-linux-servers",{"title":755,"path":756,"stem":757},"Using Docker","/deployment/using-docker","20.deployment/05.using-docker",{"title":759,"path":760,"stem":761},"Cloud Deployment Guides","/deployment/cloud-deployment-guides","20.deployment/06.cloud-deployment-guides",{"title":763,"path":764,"stem":765},"Scaling and Load Balancing","/deployment/scaling-and-load-balancing","20.deployment/07.scaling-and-load-balancing",{"title":767,"path":768,"stem":769},"Monitoring and Logging","/deployment/monitoring-and-logging","20.deployment/08.monitoring-and-logging",{"title":771,"path":772,"stem":773},"Backup Strategies","/deployment/backup-strategies","20.deployment/09.backup-strategies",{"title":775,"path":776,"stem":777,"children":778},"Performance and Optimization","/performance-and-optimization","21.performance-and-optimization/01.index",[779,780,784,788,792,796],{"title":775,"path":776,"stem":777},{"title":781,"path":782,"stem":783},"Query Optimization","/performance-and-optimization/query-optimization","21.performance-and-optimization/02.query-optimization",{"title":785,"path":786,"stem":787},"Template Rendering Optimization","/performance-and-optimization/template-rendering-optimization","21.performance-and-optimization/03.template-rendering-optimization",{"title":789,"path":790,"stem":791},"Using Select Related and Prefetch Related","/performance-and-optimization/using-select-related-and-prefetch-related","21.performance-and-optimization/04.using-select-related-and-prefetch-related",{"title":793,"path":794,"stem":795},"Caching Strategies","/performance-and-optimization/caching-strategies","21.performance-and-optimization/05.caching-strategies",{"title":797,"path":798,"stem":799},"Profiling Django Apps","/performance-and-optimization/profiling-django-apps","21.performance-and-optimization/06.profiling-django-apps",{"title":801,"path":802,"stem":803,"children":804},"Advanced and Expert Topics","/advanced-and-expert-topics","22.advanced-and-expert-topics/01.index",[805,806,810,814,818,822,826,830,834,838,842,846],{"title":801,"path":802,"stem":803},{"title":807,"path":808,"stem":809},"System Architecture Patterns","/advanced-and-expert-topics/system-architecture-patterns","22.advanced-and-expert-topics/02.system-architecture-patterns",{"title":811,"path":812,"stem":813},"Domain Driven Design with Django","/advanced-and-expert-topics/domain-driven-design-with-django","22.advanced-and-expert-topics/03.domain-driven-design-with-django",{"title":815,"path":816,"stem":817},"Building Large Scale Django Projects","/advanced-and-expert-topics/building-large-scale-django-projects","22.advanced-and-expert-topics/04.building-large-scale-django-projects",{"title":819,"path":820,"stem":821},"Plugin Architectures for Django Apps","/advanced-and-expert-topics/plugin-architectures-for-django-apps","22.advanced-and-expert-topics/05.plugin-architectures-for-django-apps",{"title":823,"path":824,"stem":825},"Extending Django's Core","/advanced-and-expert-topics/extending-djangos-core","22.advanced-and-expert-topics/06.extending-djangos-core",{"title":827,"path":828,"stem":829},"Custom ORM Expressions","/advanced-and-expert-topics/custom-orm-expressions","22.advanced-and-expert-topics/07.custom-orm-expressions",{"title":831,"path":832,"stem":833},"Custom Management Commands","/advanced-and-expert-topics/custom-management-commands","22.advanced-and-expert-topics/08.custom-management-commands",{"title":835,"path":836,"stem":837},"Working with Signals","/advanced-and-expert-topics/working-with-signals","22.advanced-and-expert-topics/09.working-with-signals",{"title":839,"path":840,"stem":841},"Building Reusable Django Packages","/advanced-and-expert-topics/building-reusable-django-packages","22.advanced-and-expert-topics/10.building-reusable-django-packages",{"title":843,"path":844,"stem":845},"Integrating Microservices","/advanced-and-expert-topics/integrating-microservices","22.advanced-and-expert-topics/11.integrating-microservices",{"title":847,"path":848,"stem":849},"Advanced Security Hardening","/advanced-and-expert-topics/advanced-security-hardening","22.advanced-and-expert-topics/12.advanced-security-hardening",{"title":851,"path":852,"stem":853,"children":854},"Logging in Django","/logging-in-django","23.logging-in-django/01.index",[855],{"title":851,"path":852,"stem":853},{"title":857,"path":858,"stem":859,"children":860},"FAQ and Troubleshooting","/faq-and-troubleshooting","24.faq-and-troubleshooting/01.index",[861],{"title":857,"path":858,"stem":859},{"title":863,"path":864,"stem":865,"children":866},"External Packages and Ecosystem","/external-packages-and-ecosystem","25.external-packages-and-ecosystem/01.index",[867],{"title":863,"path":864,"stem":865},{"title":869,"path":870,"stem":871,"children":872},"Django Internals and Contributing","/django-internals-and-contributing","26.django-internals-and-contributing/01.index",[873],{"title":869,"path":870,"stem":871},{"title":875,"path":876,"stem":877,"children":878},"Microservices with Django","/microservices-with-django","27.microservices-with-django/01.index",[879,880,884,888,892,896,900,904,908,912,916,920,924],{"title":875,"path":876,"stem":877},{"title":881,"path":882,"stem":883},"What Is a Microservice?","/microservices-with-django/what-is-a-microservice","27.microservices-with-django/02.what-is-a-microservice",{"title":885,"path":886,"stem":887},"Introducing the Django Microservices Architecture","/microservices-with-django/django-microservices-architecture","27.microservices-with-django/03.django-microservices-architecture",{"title":889,"path":890,"stem":891},"Setting Up the Development and Runtime Environment","/microservices-with-django/development-environment","27.microservices-with-django/04.development-environment",{"title":893,"path":894,"stem":895},"Cloud-native Data Processing with MongoDB","/microservices-with-django/cloud-native-data-processing","27.microservices-with-django/05.cloud-native-data-processing",{"title":897,"path":898,"stem":899},"Creating RESTful APIs for Microservices","/microservices-with-django/restful-apis","27.microservices-with-django/06.restful-apis",{"title":901,"path":902,"stem":903},"Orchestrating Microservices with Celery and RabbitMQ","/microservices-with-django/orchestrating-celery-rabbitmq","27.microservices-with-django/07.orchestrating-celery-rabbitmq",{"title":905,"path":906,"stem":907},"Testing Microservices","/microservices-with-django/testing-microservices","27.microservices-with-django/08.testing-microservices",{"title":909,"path":910,"stem":911},"Deploying Microservices","/microservices-with-django/deploying-microservices","27.microservices-with-django/09.deploying-microservices",{"title":913,"path":914,"stem":915},"Securing Microservices","/microservices-with-django/securing-microservices","27.microservices-with-django/10.securing-microservices",{"title":917,"path":918,"stem":919},"Improving Microservices Performance with Caching","/microservices-with-django/performance-caching","27.microservices-with-django/11.performance-caching",{"title":921,"path":922,"stem":923},"Best Practices","/microservices-with-django/best-practices","27.microservices-with-django/12.best-practices",{"title":925,"path":926,"stem":927},"Transforming a Monolithic Web App into a Microservice version","/microservices-with-django/monolith-to-microservices","27.microservices-with-django/13.monolith-to-microservices",{"title":929,"path":930,"stem":931,"children":932},"Releases","/releases","releases",[933,936,940,944,948,952,956,960,964,968,972,976,980,984,988,992,996,1000,1004,1008,1012],{"title":934,"path":930,"stem":935},"Django Releases","releases/index",{"title":937,"path":938,"stem":939},"Django 5.1.10 release notes","/releases/5.1.10","releases/5.1.10",{"title":941,"path":942,"stem":943},"Django 5.1.11 release notes","/releases/5.1.11","releases/5.1.11",{"title":945,"path":946,"stem":947},"Django 5.1.12 release notes","/releases/5.1.12","releases/5.1.12",{"title":949,"path":950,"stem":951},"Django 5.1.13 release notes","/releases/5.1.13","releases/5.1.13",{"title":953,"path":954,"stem":955},"Django 5.1.14 release notes","/releases/5.1.14","releases/5.1.14",{"title":957,"path":958,"stem":959},"Django 5.1.15 release notes","/releases/5.1.15","releases/5.1.15",{"title":961,"path":962,"stem":963},"Django 5.2 release notes","/releases/5.2","releases/5.2",{"title":965,"path":966,"stem":967},"Django 5.2.1 release notes","/releases/5.2.1","releases/5.2.1",{"title":969,"path":970,"stem":971},"Django 5.2.10 release notes","/releases/5.2.10","releases/5.2.10",{"title":973,"path":974,"stem":975},"Django 5.2.2 release notes","/releases/5.2.2","releases/5.2.2",{"title":977,"path":978,"stem":979},"Django 5.2.3 release notes","/releases/5.2.3","releases/5.2.3",{"title":981,"path":982,"stem":983},"Django 5.2.4 release notes","/releases/5.2.4","releases/5.2.4",{"title":985,"path":986,"stem":987},"Django 5.2.5 release notes","/releases/5.2.5","releases/5.2.5",{"title":989,"path":990,"stem":991},"Django 5.2.6 release notes","/releases/5.2.6","releases/5.2.6",{"title":993,"path":994,"stem":995},"Django 5.2.7 release notes","/releases/5.2.7","releases/5.2.7",{"title":997,"path":998,"stem":999},"Django 5.2.8 release notes","/releases/5.2.8","releases/5.2.8",{"title":1001,"path":1002,"stem":1003},"Django 5.2.9 release notes","/releases/5.2.9","releases/5.2.9",{"title":1005,"path":1006,"stem":1007},"Django 6.0 release notes","/releases/6.0","releases/6.0",{"title":1009,"path":1010,"stem":1011},"Django 6.0.1 release notes","/releases/6.0.1","releases/6.0.1",{"title":1013,"path":1014,"stem":1015},"Django 6.1 release notes - UNDER DEVELOPMENT","/releases/6.1","releases/6.1",{"id":1017,"title":511,"body":1018,"description":1028,"extension":12040,"links":12041,"meta":12042,"navigation":1074,"path":512,"seo":12043,"stem":513,"__hash__":12044},"docs/12.admin-site/06.admin-security-best-practices.md",{"type":1019,"value":1020,"toc":12006},"minimark",[1021,1025,1029,1034,1039,1406,1410,1846,1850,2843,2847,2851,3216,3220,4258,4262,5298,5302,5306,6043,6047,7444,7447,7451,8892,8896,9865,9869,9873,10455,10459,10778,10782,10786,11906,11910,11914,11928,11932,11946,11950,11964,11968,11982,11986,12000,12003],[1022,1023,511],"h1",{"id":1024},"admin-security-best-practices",[1026,1027,1028],"p",{},"Securing the Django admin interface is crucial for protecting your application and data. This chapter covers comprehensive security measures, from basic configurations to advanced protection strategies.",[1030,1031,1033],"h2",{"id":1032},"authentication-and-access-control","Authentication and Access Control",[1035,1036,1038],"h3",{"id":1037},"strong-authentication-requirements","Strong Authentication Requirements",[1040,1041,1046],"pre",{"className":1042,"code":1043,"language":1044,"meta":1045,"style":1045},"language-python shiki shiki-themes material-theme-lighter vitesse-light vitesse-dark","# settings.py\nimport os\n\n# Password validation\nAUTH_PASSWORD_VALIDATORS = [\n    {\n        'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator',\n    },\n    {\n        'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator',\n        'OPTIONS': {\n            'min_length': 12,  # Require longer passwords\n        }\n    },\n    {\n        'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator',\n    },\n    {\n        'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator',\n    },\n]\n\n# Session security\nSESSION_COOKIE_SECURE = True  # HTTPS only\nSESSION_COOKIE_HTTPONLY = True  # No JavaScript access\nSESSION_COOKIE_AGE = 3600  # 1 hour timeout\nSESSION_EXPIRE_AT_BROWSER_CLOSE = True\n\n# CSRF protection\nCSRF_COOKIE_SECURE = True\nCSRF_COOKIE_HTTPONLY = True\nCSRF_USE_SESSIONS = True\n\n# Admin-specific settings\nADMIN_SESSION_TIMEOUT = 1800  # 30 minutes\n","python","",[1047,1048,1049,1058,1069,1076,1082,1096,1102,1130,1136,1141,1161,1176,1199,1205,1210,1215,1235,1240,1245,1265,1270,1276,1281,1287,1302,1315,1329,1340,1345,1351,1361,1371,1381,1386,1392],"code",{"__ignoreMap":1045},[1050,1051,1054],"span",{"class":1052,"line":1053},"line",1,[1050,1055,1057],{"class":1056},"s9Tkl","# settings.py\n",[1050,1059,1061,1065],{"class":1052,"line":1060},2,[1050,1062,1064],{"class":1063},"siDh9","import",[1050,1066,1068],{"class":1067},"sftqT"," os\n",[1050,1070,1072],{"class":1052,"line":1071},3,[1050,1073,1075],{"emptyLinePlaceholder":1074},true,"\n",[1050,1077,1079],{"class":1052,"line":1078},4,[1050,1080,1081],{"class":1056},"# Password validation\n",[1050,1083,1085,1089,1093],{"class":1052,"line":1084},5,[1050,1086,1088],{"class":1087},"se3Ec","AUTH_PASSWORD_VALIDATORS",[1050,1090,1092],{"class":1091},"soVBu"," =",[1050,1094,1095],{"class":1091}," [\n",[1050,1097,1099],{"class":1052,"line":1098},6,[1050,1100,1101],{"class":1091},"    {\n",[1050,1103,1105,1109,1113,1116,1119,1122,1125,1127],{"class":1052,"line":1104},7,[1050,1106,1108],{"class":1107},"sbYkP","        '",[1050,1110,1112],{"class":1111},"sTbE_","NAME",[1050,1114,1115],{"class":1107},"'",[1050,1117,1118],{"class":1091},":",[1050,1120,1121],{"class":1107}," '",[1050,1123,1124],{"class":1111},"django.contrib.auth.password_validation.UserAttributeSimilarityValidator",[1050,1126,1115],{"class":1107},[1050,1128,1129],{"class":1091},",\n",[1050,1131,1133],{"class":1052,"line":1132},8,[1050,1134,1135],{"class":1091},"    },\n",[1050,1137,1139],{"class":1052,"line":1138},9,[1050,1140,1101],{"class":1091},[1050,1142,1144,1146,1148,1150,1152,1154,1157,1159],{"class":1052,"line":1143},10,[1050,1145,1108],{"class":1107},[1050,1147,1112],{"class":1111},[1050,1149,1115],{"class":1107},[1050,1151,1118],{"class":1091},[1050,1153,1121],{"class":1107},[1050,1155,1156],{"class":1111},"django.contrib.auth.password_validation.MinimumLengthValidator",[1050,1158,1115],{"class":1107},[1050,1160,1129],{"class":1091},[1050,1162,1164,1166,1169,1171,1173],{"class":1052,"line":1163},11,[1050,1165,1108],{"class":1107},[1050,1167,1168],{"class":1111},"OPTIONS",[1050,1170,1115],{"class":1107},[1050,1172,1118],{"class":1091},[1050,1174,1175],{"class":1091}," {\n",[1050,1177,1179,1182,1185,1187,1189,1193,1196],{"class":1052,"line":1178},12,[1050,1180,1181],{"class":1107},"            '",[1050,1183,1184],{"class":1111},"min_length",[1050,1186,1115],{"class":1107},[1050,1188,1118],{"class":1091},[1050,1190,1192],{"class":1191},"s7CZa"," 12",[1050,1194,1195],{"class":1091},",",[1050,1197,1198],{"class":1056},"  # Require longer passwords\n",[1050,1200,1202],{"class":1052,"line":1201},13,[1050,1203,1204],{"class":1091},"        }\n",[1050,1206,1208],{"class":1052,"line":1207},14,[1050,1209,1135],{"class":1091},[1050,1211,1213],{"class":1052,"line":1212},15,[1050,1214,1101],{"class":1091},[1050,1216,1218,1220,1222,1224,1226,1228,1231,1233],{"class":1052,"line":1217},16,[1050,1219,1108],{"class":1107},[1050,1221,1112],{"class":1111},[1050,1223,1115],{"class":1107},[1050,1225,1118],{"class":1091},[1050,1227,1121],{"class":1107},[1050,1229,1230],{"class":1111},"django.contrib.auth.password_validation.CommonPasswordValidator",[1050,1232,1115],{"class":1107},[1050,1234,1129],{"class":1091},[1050,1236,1238],{"class":1052,"line":1237},17,[1050,1239,1135],{"class":1091},[1050,1241,1243],{"class":1052,"line":1242},18,[1050,1244,1101],{"class":1091},[1050,1246,1248,1250,1252,1254,1256,1258,1261,1263],{"class":1052,"line":1247},19,[1050,1249,1108],{"class":1107},[1050,1251,1112],{"class":1111},[1050,1253,1115],{"class":1107},[1050,1255,1118],{"class":1091},[1050,1257,1121],{"class":1107},[1050,1259,1260],{"class":1111},"django.contrib.auth.password_validation.NumericPasswordValidator",[1050,1262,1115],{"class":1107},[1050,1264,1129],{"class":1091},[1050,1266,1268],{"class":1052,"line":1267},20,[1050,1269,1135],{"class":1091},[1050,1271,1273],{"class":1052,"line":1272},21,[1050,1274,1275],{"class":1091},"]\n",[1050,1277,1279],{"class":1052,"line":1278},22,[1050,1280,1075],{"emptyLinePlaceholder":1074},[1050,1282,1284],{"class":1052,"line":1283},23,[1050,1285,1286],{"class":1056},"# Session security\n",[1050,1288,1290,1293,1295,1299],{"class":1052,"line":1289},24,[1050,1291,1292],{"class":1087},"SESSION_COOKIE_SECURE",[1050,1294,1092],{"class":1091},[1050,1296,1298],{"class":1297},"s8XtY"," True",[1050,1300,1301],{"class":1056},"  # HTTPS only\n",[1050,1303,1305,1308,1310,1312],{"class":1052,"line":1304},25,[1050,1306,1307],{"class":1087},"SESSION_COOKIE_HTTPONLY",[1050,1309,1092],{"class":1091},[1050,1311,1298],{"class":1297},[1050,1313,1314],{"class":1056},"  # No JavaScript access\n",[1050,1316,1318,1321,1323,1326],{"class":1052,"line":1317},26,[1050,1319,1320],{"class":1087},"SESSION_COOKIE_AGE",[1050,1322,1092],{"class":1091},[1050,1324,1325],{"class":1191}," 3600",[1050,1327,1328],{"class":1056},"  # 1 hour timeout\n",[1050,1330,1332,1335,1337],{"class":1052,"line":1331},27,[1050,1333,1334],{"class":1087},"SESSION_EXPIRE_AT_BROWSER_CLOSE",[1050,1336,1092],{"class":1091},[1050,1338,1339],{"class":1297}," True\n",[1050,1341,1343],{"class":1052,"line":1342},28,[1050,1344,1075],{"emptyLinePlaceholder":1074},[1050,1346,1348],{"class":1052,"line":1347},29,[1050,1349,1350],{"class":1056},"# CSRF protection\n",[1050,1352,1354,1357,1359],{"class":1052,"line":1353},30,[1050,1355,1356],{"class":1087},"CSRF_COOKIE_SECURE",[1050,1358,1092],{"class":1091},[1050,1360,1339],{"class":1297},[1050,1362,1364,1367,1369],{"class":1052,"line":1363},31,[1050,1365,1366],{"class":1087},"CSRF_COOKIE_HTTPONLY",[1050,1368,1092],{"class":1091},[1050,1370,1339],{"class":1297},[1050,1372,1374,1377,1379],{"class":1052,"line":1373},32,[1050,1375,1376],{"class":1087},"CSRF_USE_SESSIONS",[1050,1378,1092],{"class":1091},[1050,1380,1339],{"class":1297},[1050,1382,1384],{"class":1052,"line":1383},33,[1050,1385,1075],{"emptyLinePlaceholder":1074},[1050,1387,1389],{"class":1052,"line":1388},34,[1050,1390,1391],{"class":1056},"# Admin-specific settings\n",[1050,1393,1395,1398,1400,1403],{"class":1052,"line":1394},35,[1050,1396,1397],{"class":1087},"ADMIN_SESSION_TIMEOUT",[1050,1399,1092],{"class":1091},[1050,1401,1402],{"class":1191}," 1800",[1050,1404,1405],{"class":1056},"  # 30 minutes\n",[1035,1407,1409],{"id":1408},"two-factor-authentication","Two-Factor Authentication",[1040,1411,1413],{"className":1042,"code":1412,"language":1044,"meta":1045,"style":1045},"# Install: pip install django-otp qrcode\n\n# settings.py\nINSTALLED_APPS = [\n    'django_otp',\n    'django_otp.plugins.otp_totp',\n    'django_otp.plugins.otp_static',\n    # ... other apps\n]\n\nMIDDLEWARE = [\n    'django_otp.middleware.OTPMiddleware',\n    # ... other middleware\n]\n\n# Custom admin site with 2FA\nfrom django_otp.admin import OTPAdminSite\nfrom django_otp.decorators import otp_required\n\nclass SecureAdminSite(OTPAdminSite):\n    \"\"\"Admin site with mandatory 2FA\"\"\"\n    \n    def has_permission(self, request):\n        \"\"\"Require 2FA for admin access\"\"\"\n        return (\n            request.user.is_active and\n            request.user.is_staff and\n            request.user.is_verified()\n        )\n\n# Replace default admin site\nadmin_site = SecureAdminSite(name='secure_admin')\n\n# Register models with secure admin\nfrom django.contrib.auth.models import User, Group\nfrom django_otp.admin import OTPTokenAdmin\nfrom django_otp.models import Device\n\nadmin_site.register(User)\nadmin_site.register(Group)\n",[1047,1414,1415,1420,1424,1428,1437,1449,1460,1471,1476,1480,1484,1493,1504,1509,1513,1517,1522,1541,1557,1561,1581,1594,1599,1622,1632,1640,1660,1675,1692,1697,1701,1706,1734,1738,1743,1775,1791,1807,1812,1830],{"__ignoreMap":1045},[1050,1416,1417],{"class":1052,"line":1053},[1050,1418,1419],{"class":1056},"# Install: pip install django-otp qrcode\n",[1050,1421,1422],{"class":1052,"line":1060},[1050,1423,1075],{"emptyLinePlaceholder":1074},[1050,1425,1426],{"class":1052,"line":1071},[1050,1427,1057],{"class":1056},[1050,1429,1430,1433,1435],{"class":1052,"line":1078},[1050,1431,1432],{"class":1087},"INSTALLED_APPS",[1050,1434,1092],{"class":1091},[1050,1436,1095],{"class":1091},[1050,1438,1439,1442,1445,1447],{"class":1052,"line":1084},[1050,1440,1441],{"class":1107},"    '",[1050,1443,1444],{"class":1111},"django_otp",[1050,1446,1115],{"class":1107},[1050,1448,1129],{"class":1091},[1050,1450,1451,1453,1456,1458],{"class":1052,"line":1098},[1050,1452,1441],{"class":1107},[1050,1454,1455],{"class":1111},"django_otp.plugins.otp_totp",[1050,1457,1115],{"class":1107},[1050,1459,1129],{"class":1091},[1050,1461,1462,1464,1467,1469],{"class":1052,"line":1104},[1050,1463,1441],{"class":1107},[1050,1465,1466],{"class":1111},"django_otp.plugins.otp_static",[1050,1468,1115],{"class":1107},[1050,1470,1129],{"class":1091},[1050,1472,1473],{"class":1052,"line":1132},[1050,1474,1475],{"class":1056},"    # ... other apps\n",[1050,1477,1478],{"class":1052,"line":1138},[1050,1479,1275],{"class":1091},[1050,1481,1482],{"class":1052,"line":1143},[1050,1483,1075],{"emptyLinePlaceholder":1074},[1050,1485,1486,1489,1491],{"class":1052,"line":1163},[1050,1487,1488],{"class":1087},"MIDDLEWARE",[1050,1490,1092],{"class":1091},[1050,1492,1095],{"class":1091},[1050,1494,1495,1497,1500,1502],{"class":1052,"line":1178},[1050,1496,1441],{"class":1107},[1050,1498,1499],{"class":1111},"django_otp.middleware.OTPMiddleware",[1050,1501,1115],{"class":1107},[1050,1503,1129],{"class":1091},[1050,1505,1506],{"class":1052,"line":1201},[1050,1507,1508],{"class":1056},"    # ... other middleware\n",[1050,1510,1511],{"class":1052,"line":1207},[1050,1512,1275],{"class":1091},[1050,1514,1515],{"class":1052,"line":1212},[1050,1516,1075],{"emptyLinePlaceholder":1074},[1050,1518,1519],{"class":1052,"line":1217},[1050,1520,1521],{"class":1056},"# Custom admin site with 2FA\n",[1050,1523,1524,1527,1530,1533,1536,1538],{"class":1052,"line":1237},[1050,1525,1526],{"class":1063},"from",[1050,1528,1529],{"class":1067}," django_otp",[1050,1531,1532],{"class":1091},".",[1050,1534,1535],{"class":1067},"admin ",[1050,1537,1064],{"class":1063},[1050,1539,1540],{"class":1067}," OTPAdminSite\n",[1050,1542,1543,1545,1547,1549,1552,1554],{"class":1052,"line":1242},[1050,1544,1526],{"class":1063},[1050,1546,1529],{"class":1067},[1050,1548,1532],{"class":1091},[1050,1550,1551],{"class":1067},"decorators ",[1050,1553,1064],{"class":1063},[1050,1555,1556],{"class":1067}," otp_required\n",[1050,1558,1559],{"class":1052,"line":1247},[1050,1560,1075],{"emptyLinePlaceholder":1074},[1050,1562,1563,1567,1571,1574,1578],{"class":1052,"line":1267},[1050,1564,1566],{"class":1565},"s5Kfy","class",[1050,1568,1570],{"class":1569},"sD-vU"," SecureAdminSite",[1050,1572,1573],{"class":1091},"(",[1050,1575,1577],{"class":1576},"sYn-s","OTPAdminSite",[1050,1579,1580],{"class":1091},"):\n",[1050,1582,1583,1587,1591],{"class":1052,"line":1272},[1050,1584,1586],{"class":1585},"sm7ve","    \"\"\"",[1050,1588,1590],{"class":1589},"sVyVU","Admin site with mandatory 2FA",[1050,1592,1593],{"class":1585},"\"\"\"\n",[1050,1595,1596],{"class":1052,"line":1278},[1050,1597,1598],{"class":1067},"    \n",[1050,1600,1601,1604,1608,1610,1614,1616,1620],{"class":1052,"line":1283},[1050,1602,1603],{"class":1565},"    def",[1050,1605,1607],{"class":1606},"sljsM"," has_permission",[1050,1609,1573],{"class":1091},[1050,1611,1613],{"class":1612},"sRjD_","self",[1050,1615,1195],{"class":1091},[1050,1617,1619],{"class":1618},"sCyAa"," request",[1050,1621,1580],{"class":1091},[1050,1623,1624,1627,1630],{"class":1052,"line":1289},[1050,1625,1626],{"class":1585},"        \"\"\"",[1050,1628,1629],{"class":1589},"Require 2FA for admin access",[1050,1631,1593],{"class":1585},[1050,1633,1634,1637],{"class":1052,"line":1304},[1050,1635,1636],{"class":1063},"        return",[1050,1638,1639],{"class":1091}," (\n",[1050,1641,1642,1645,1647,1651,1653,1656],{"class":1052,"line":1317},[1050,1643,1644],{"class":1067},"            request",[1050,1646,1532],{"class":1091},[1050,1648,1650],{"class":1649},"sBPpx","user",[1050,1652,1532],{"class":1091},[1050,1654,1655],{"class":1649},"is_active",[1050,1657,1659],{"class":1658},"sVsLi"," and\n",[1050,1661,1662,1664,1666,1668,1670,1673],{"class":1052,"line":1331},[1050,1663,1644],{"class":1067},[1050,1665,1532],{"class":1091},[1050,1667,1650],{"class":1649},[1050,1669,1532],{"class":1091},[1050,1671,1672],{"class":1649},"is_staff",[1050,1674,1659],{"class":1658},[1050,1676,1677,1679,1681,1683,1685,1689],{"class":1052,"line":1342},[1050,1678,1644],{"class":1067},[1050,1680,1532],{"class":1091},[1050,1682,1650],{"class":1649},[1050,1684,1532],{"class":1091},[1050,1686,1688],{"class":1687},"siWMO","is_verified",[1050,1690,1691],{"class":1091},"()\n",[1050,1693,1694],{"class":1052,"line":1347},[1050,1695,1696],{"class":1091},"        )\n",[1050,1698,1699],{"class":1052,"line":1353},[1050,1700,1075],{"emptyLinePlaceholder":1074},[1050,1702,1703],{"class":1052,"line":1363},[1050,1704,1705],{"class":1056},"# Replace default admin site\n",[1050,1707,1708,1711,1714,1716,1718,1722,1724,1726,1729,1731],{"class":1052,"line":1373},[1050,1709,1710],{"class":1067},"admin_site ",[1050,1712,1713],{"class":1091},"=",[1050,1715,1570],{"class":1687},[1050,1717,1573],{"class":1091},[1050,1719,1721],{"class":1720},"sqOPj","name",[1050,1723,1713],{"class":1091},[1050,1725,1115],{"class":1107},[1050,1727,1728],{"class":1111},"secure_admin",[1050,1730,1115],{"class":1107},[1050,1732,1733],{"class":1091},")\n",[1050,1735,1736],{"class":1052,"line":1383},[1050,1737,1075],{"emptyLinePlaceholder":1074},[1050,1739,1740],{"class":1052,"line":1388},[1050,1741,1742],{"class":1056},"# Register models with secure admin\n",[1050,1744,1745,1747,1750,1752,1755,1757,1760,1762,1765,1767,1770,1772],{"class":1052,"line":1394},[1050,1746,1526],{"class":1063},[1050,1748,1749],{"class":1067}," django",[1050,1751,1532],{"class":1091},[1050,1753,1754],{"class":1067},"contrib",[1050,1756,1532],{"class":1091},[1050,1758,1759],{"class":1067},"auth",[1050,1761,1532],{"class":1091},[1050,1763,1764],{"class":1067},"models ",[1050,1766,1064],{"class":1063},[1050,1768,1769],{"class":1067}," User",[1050,1771,1195],{"class":1091},[1050,1773,1774],{"class":1067}," Group\n",[1050,1776,1778,1780,1782,1784,1786,1788],{"class":1052,"line":1777},36,[1050,1779,1526],{"class":1063},[1050,1781,1529],{"class":1067},[1050,1783,1532],{"class":1091},[1050,1785,1535],{"class":1067},[1050,1787,1064],{"class":1063},[1050,1789,1790],{"class":1067}," OTPTokenAdmin\n",[1050,1792,1794,1796,1798,1800,1802,1804],{"class":1052,"line":1793},37,[1050,1795,1526],{"class":1063},[1050,1797,1529],{"class":1067},[1050,1799,1532],{"class":1091},[1050,1801,1764],{"class":1067},[1050,1803,1064],{"class":1063},[1050,1805,1806],{"class":1067}," Device\n",[1050,1808,1810],{"class":1052,"line":1809},38,[1050,1811,1075],{"emptyLinePlaceholder":1074},[1050,1813,1815,1818,1820,1823,1825,1828],{"class":1052,"line":1814},39,[1050,1816,1817],{"class":1067},"admin_site",[1050,1819,1532],{"class":1091},[1050,1821,1822],{"class":1687},"register",[1050,1824,1573],{"class":1091},[1050,1826,1827],{"class":1687},"User",[1050,1829,1733],{"class":1091},[1050,1831,1833,1835,1837,1839,1841,1844],{"class":1052,"line":1832},40,[1050,1834,1817],{"class":1067},[1050,1836,1532],{"class":1091},[1050,1838,1822],{"class":1687},[1050,1840,1573],{"class":1091},[1050,1842,1843],{"class":1687},"Group",[1050,1845,1733],{"class":1091},[1035,1847,1849],{"id":1848},"role-based-access-control","Role-Based Access Control",[1040,1851,1853],{"className":1042,"code":1852,"language":1044,"meta":1045,"style":1045},"# models.py\nfrom django.contrib.auth.models import AbstractUser, Group, Permission\nfrom django.db import models\n\nclass CustomUser(AbstractUser):\n    \"\"\"Extended user model with admin roles\"\"\"\n    \n    ADMIN_ROLES = [\n        ('content_admin', 'Content Administrator'),\n        ('user_admin', 'User Administrator'),\n        ('system_admin', 'System Administrator'),\n        ('read_only', 'Read Only Access'),\n    ]\n    \n    admin_role = models.CharField(\n        max_length=20,\n        choices=ADMIN_ROLES,\n        blank=True,\n        null=True\n    )\n    \n    last_admin_login = models.DateTimeField(null=True, blank=True)\n    failed_login_attempts = models.IntegerField(default=0)\n    account_locked_until = models.DateTimeField(null=True, blank=True)\n\n# admin.py\nfrom django.contrib import admin\nfrom django.contrib.auth.admin import UserAdmin\nfrom django.utils import timezone\n\nclass SecureUserAdmin(UserAdmin):\n    \"\"\"Secure user admin with role management\"\"\"\n    \n    list_display = [\n        'username', 'email', 'admin_role', 'is_staff', \n        'last_admin_login', 'failed_login_attempts'\n    ]\n    \n    list_filter = ['admin_role', 'is_staff', 'is_superuser', 'last_admin_login']\n    \n    fieldsets = UserAdmin.fieldsets + (\n        ('Admin Security', {\n            'fields': (\n                'admin_role', \n                'last_admin_login', \n                'failed_login_attempts',\n                'account_locked_until'\n            )\n        }),\n    )\n    \n    readonly_fields = ['last_admin_login', 'failed_login_attempts']\n    \n    def get_queryset(self, request):\n        \"\"\"Filter users based on admin role\"\"\"\n        queryset = super().get_queryset(request)\n        \n        if not request.user.is_superuser:\n            # Non-superusers can only see users with lower privileges\n            if request.user.admin_role == 'user_admin':\n                queryset = queryset.filter(\n                    admin_role__in=['content_admin', 'read_only']\n                )\n            elif request.user.admin_role == 'content_admin':\n                queryset = queryset.filter(admin_role='read_only')\n        \n        return queryset\n\nadmin.site.register(CustomUser, SecureUserAdmin)\n",[1047,1854,1855,1860,1893,1909,1913,1927,1936,1940,1949,1973,1995,2017,2039,2044,2048,2066,2078,2091,2103,2113,2118,2122,2156,2182,2213,2217,2222,2238,2261,2277,2281,2295,2304,2308,2317,2357,2376,2380,2384,2427,2431,2452,2468,2482,2496,2509,2520,2530,2536,2542,2547,2552,2578,2583,2601,2611,2636,2642,2664,2670,2697,2715,2740,2746,2772,2799,2804,2812,2817],{"__ignoreMap":1045},[1050,1856,1857],{"class":1052,"line":1053},[1050,1858,1859],{"class":1056},"# models.py\n",[1050,1861,1862,1864,1866,1868,1870,1872,1874,1876,1878,1880,1883,1885,1888,1890],{"class":1052,"line":1060},[1050,1863,1526],{"class":1063},[1050,1865,1749],{"class":1067},[1050,1867,1532],{"class":1091},[1050,1869,1754],{"class":1067},[1050,1871,1532],{"class":1091},[1050,1873,1759],{"class":1067},[1050,1875,1532],{"class":1091},[1050,1877,1764],{"class":1067},[1050,1879,1064],{"class":1063},[1050,1881,1882],{"class":1067}," AbstractUser",[1050,1884,1195],{"class":1091},[1050,1886,1887],{"class":1067}," Group",[1050,1889,1195],{"class":1091},[1050,1891,1892],{"class":1067}," Permission\n",[1050,1894,1895,1897,1899,1901,1904,1906],{"class":1052,"line":1071},[1050,1896,1526],{"class":1063},[1050,1898,1749],{"class":1067},[1050,1900,1532],{"class":1091},[1050,1902,1903],{"class":1067},"db ",[1050,1905,1064],{"class":1063},[1050,1907,1908],{"class":1067}," models\n",[1050,1910,1911],{"class":1052,"line":1078},[1050,1912,1075],{"emptyLinePlaceholder":1074},[1050,1914,1915,1917,1920,1922,1925],{"class":1052,"line":1084},[1050,1916,1566],{"class":1565},[1050,1918,1919],{"class":1569}," CustomUser",[1050,1921,1573],{"class":1091},[1050,1923,1924],{"class":1576},"AbstractUser",[1050,1926,1580],{"class":1091},[1050,1928,1929,1931,1934],{"class":1052,"line":1098},[1050,1930,1586],{"class":1585},[1050,1932,1933],{"class":1589},"Extended user model with admin roles",[1050,1935,1593],{"class":1585},[1050,1937,1938],{"class":1052,"line":1104},[1050,1939,1598],{"class":1067},[1050,1941,1942,1945,1947],{"class":1052,"line":1132},[1050,1943,1944],{"class":1087},"    ADMIN_ROLES",[1050,1946,1092],{"class":1091},[1050,1948,1095],{"class":1091},[1050,1950,1951,1954,1956,1959,1961,1963,1965,1968,1970],{"class":1052,"line":1138},[1050,1952,1953],{"class":1091},"        (",[1050,1955,1115],{"class":1107},[1050,1957,1958],{"class":1111},"content_admin",[1050,1960,1115],{"class":1107},[1050,1962,1195],{"class":1091},[1050,1964,1121],{"class":1107},[1050,1966,1967],{"class":1111},"Content Administrator",[1050,1969,1115],{"class":1107},[1050,1971,1972],{"class":1091},"),\n",[1050,1974,1975,1977,1979,1982,1984,1986,1988,1991,1993],{"class":1052,"line":1143},[1050,1976,1953],{"class":1091},[1050,1978,1115],{"class":1107},[1050,1980,1981],{"class":1111},"user_admin",[1050,1983,1115],{"class":1107},[1050,1985,1195],{"class":1091},[1050,1987,1121],{"class":1107},[1050,1989,1990],{"class":1111},"User Administrator",[1050,1992,1115],{"class":1107},[1050,1994,1972],{"class":1091},[1050,1996,1997,1999,2001,2004,2006,2008,2010,2013,2015],{"class":1052,"line":1163},[1050,1998,1953],{"class":1091},[1050,2000,1115],{"class":1107},[1050,2002,2003],{"class":1111},"system_admin",[1050,2005,1115],{"class":1107},[1050,2007,1195],{"class":1091},[1050,2009,1121],{"class":1107},[1050,2011,2012],{"class":1111},"System Administrator",[1050,2014,1115],{"class":1107},[1050,2016,1972],{"class":1091},[1050,2018,2019,2021,2023,2026,2028,2030,2032,2035,2037],{"class":1052,"line":1178},[1050,2020,1953],{"class":1091},[1050,2022,1115],{"class":1107},[1050,2024,2025],{"class":1111},"read_only",[1050,2027,1115],{"class":1107},[1050,2029,1195],{"class":1091},[1050,2031,1121],{"class":1107},[1050,2033,2034],{"class":1111},"Read Only Access",[1050,2036,1115],{"class":1107},[1050,2038,1972],{"class":1091},[1050,2040,2041],{"class":1052,"line":1201},[1050,2042,2043],{"class":1091},"    ]\n",[1050,2045,2046],{"class":1052,"line":1207},[1050,2047,1598],{"class":1067},[1050,2049,2050,2053,2055,2058,2060,2063],{"class":1052,"line":1212},[1050,2051,2052],{"class":1067},"    admin_role ",[1050,2054,1713],{"class":1091},[1050,2056,2057],{"class":1067}," models",[1050,2059,1532],{"class":1091},[1050,2061,2062],{"class":1687},"CharField",[1050,2064,2065],{"class":1091},"(\n",[1050,2067,2068,2071,2073,2076],{"class":1052,"line":1217},[1050,2069,2070],{"class":1720},"        max_length",[1050,2072,1713],{"class":1091},[1050,2074,2075],{"class":1191},"20",[1050,2077,1129],{"class":1091},[1050,2079,2080,2083,2085,2089],{"class":1052,"line":1237},[1050,2081,2082],{"class":1720},"        choices",[1050,2084,1713],{"class":1091},[1050,2086,2088],{"class":2087},"s6DHA","ADMIN_ROLES",[1050,2090,1129],{"class":1091},[1050,2092,2093,2096,2098,2101],{"class":1052,"line":1242},[1050,2094,2095],{"class":1720},"        blank",[1050,2097,1713],{"class":1091},[1050,2099,2100],{"class":1297},"True",[1050,2102,1129],{"class":1091},[1050,2104,2105,2108,2110],{"class":1052,"line":1247},[1050,2106,2107],{"class":1720},"        null",[1050,2109,1713],{"class":1091},[1050,2111,2112],{"class":1297},"True\n",[1050,2114,2115],{"class":1052,"line":1267},[1050,2116,2117],{"class":1091},"    )\n",[1050,2119,2120],{"class":1052,"line":1272},[1050,2121,1598],{"class":1067},[1050,2123,2124,2127,2129,2131,2133,2136,2138,2141,2143,2145,2147,2150,2152,2154],{"class":1052,"line":1278},[1050,2125,2126],{"class":1067},"    last_admin_login ",[1050,2128,1713],{"class":1091},[1050,2130,2057],{"class":1067},[1050,2132,1532],{"class":1091},[1050,2134,2135],{"class":1687},"DateTimeField",[1050,2137,1573],{"class":1091},[1050,2139,2140],{"class":1720},"null",[1050,2142,1713],{"class":1091},[1050,2144,2100],{"class":1297},[1050,2146,1195],{"class":1091},[1050,2148,2149],{"class":1720}," blank",[1050,2151,1713],{"class":1091},[1050,2153,2100],{"class":1297},[1050,2155,1733],{"class":1091},[1050,2157,2158,2161,2163,2165,2167,2170,2172,2175,2177,2180],{"class":1052,"line":1283},[1050,2159,2160],{"class":1067},"    failed_login_attempts ",[1050,2162,1713],{"class":1091},[1050,2164,2057],{"class":1067},[1050,2166,1532],{"class":1091},[1050,2168,2169],{"class":1687},"IntegerField",[1050,2171,1573],{"class":1091},[1050,2173,2174],{"class":1720},"default",[1050,2176,1713],{"class":1091},[1050,2178,2179],{"class":1191},"0",[1050,2181,1733],{"class":1091},[1050,2183,2184,2187,2189,2191,2193,2195,2197,2199,2201,2203,2205,2207,2209,2211],{"class":1052,"line":1289},[1050,2185,2186],{"class":1067},"    account_locked_until ",[1050,2188,1713],{"class":1091},[1050,2190,2057],{"class":1067},[1050,2192,1532],{"class":1091},[1050,2194,2135],{"class":1687},[1050,2196,1573],{"class":1091},[1050,2198,2140],{"class":1720},[1050,2200,1713],{"class":1091},[1050,2202,2100],{"class":1297},[1050,2204,1195],{"class":1091},[1050,2206,2149],{"class":1720},[1050,2208,1713],{"class":1091},[1050,2210,2100],{"class":1297},[1050,2212,1733],{"class":1091},[1050,2214,2215],{"class":1052,"line":1304},[1050,2216,1075],{"emptyLinePlaceholder":1074},[1050,2218,2219],{"class":1052,"line":1317},[1050,2220,2221],{"class":1056},"# admin.py\n",[1050,2223,2224,2226,2228,2230,2233,2235],{"class":1052,"line":1331},[1050,2225,1526],{"class":1063},[1050,2227,1749],{"class":1067},[1050,2229,1532],{"class":1091},[1050,2231,2232],{"class":1067},"contrib ",[1050,2234,1064],{"class":1063},[1050,2236,2237],{"class":1067}," admin\n",[1050,2239,2240,2242,2244,2246,2248,2250,2252,2254,2256,2258],{"class":1052,"line":1342},[1050,2241,1526],{"class":1063},[1050,2243,1749],{"class":1067},[1050,2245,1532],{"class":1091},[1050,2247,1754],{"class":1067},[1050,2249,1532],{"class":1091},[1050,2251,1759],{"class":1067},[1050,2253,1532],{"class":1091},[1050,2255,1535],{"class":1067},[1050,2257,1064],{"class":1063},[1050,2259,2260],{"class":1067}," UserAdmin\n",[1050,2262,2263,2265,2267,2269,2272,2274],{"class":1052,"line":1347},[1050,2264,1526],{"class":1063},[1050,2266,1749],{"class":1067},[1050,2268,1532],{"class":1091},[1050,2270,2271],{"class":1067},"utils ",[1050,2273,1064],{"class":1063},[1050,2275,2276],{"class":1067}," timezone\n",[1050,2278,2279],{"class":1052,"line":1353},[1050,2280,1075],{"emptyLinePlaceholder":1074},[1050,2282,2283,2285,2288,2290,2293],{"class":1052,"line":1363},[1050,2284,1566],{"class":1565},[1050,2286,2287],{"class":1569}," SecureUserAdmin",[1050,2289,1573],{"class":1091},[1050,2291,2292],{"class":1576},"UserAdmin",[1050,2294,1580],{"class":1091},[1050,2296,2297,2299,2302],{"class":1052,"line":1373},[1050,2298,1586],{"class":1585},[1050,2300,2301],{"class":1589},"Secure user admin with role management",[1050,2303,1593],{"class":1585},[1050,2305,2306],{"class":1052,"line":1383},[1050,2307,1598],{"class":1067},[1050,2309,2310,2313,2315],{"class":1052,"line":1388},[1050,2311,2312],{"class":1067},"    list_display ",[1050,2314,1713],{"class":1091},[1050,2316,1095],{"class":1091},[1050,2318,2319,2321,2324,2326,2328,2330,2333,2335,2337,2339,2342,2344,2346,2348,2350,2352,2354],{"class":1052,"line":1394},[1050,2320,1108],{"class":1107},[1050,2322,2323],{"class":1111},"username",[1050,2325,1115],{"class":1107},[1050,2327,1195],{"class":1091},[1050,2329,1121],{"class":1107},[1050,2331,2332],{"class":1111},"email",[1050,2334,1115],{"class":1107},[1050,2336,1195],{"class":1091},[1050,2338,1121],{"class":1107},[1050,2340,2341],{"class":1111},"admin_role",[1050,2343,1115],{"class":1107},[1050,2345,1195],{"class":1091},[1050,2347,1121],{"class":1107},[1050,2349,1672],{"class":1111},[1050,2351,1115],{"class":1107},[1050,2353,1195],{"class":1091},[1050,2355,2356],{"class":1067}," \n",[1050,2358,2359,2361,2364,2366,2368,2370,2373],{"class":1052,"line":1777},[1050,2360,1108],{"class":1107},[1050,2362,2363],{"class":1111},"last_admin_login",[1050,2365,1115],{"class":1107},[1050,2367,1195],{"class":1091},[1050,2369,1121],{"class":1107},[1050,2371,2372],{"class":1111},"failed_login_attempts",[1050,2374,2375],{"class":1107},"'\n",[1050,2377,2378],{"class":1052,"line":1793},[1050,2379,2043],{"class":1091},[1050,2381,2382],{"class":1052,"line":1809},[1050,2383,1598],{"class":1067},[1050,2385,2386,2389,2391,2394,2396,2398,2400,2402,2404,2406,2408,2410,2412,2415,2417,2419,2421,2423,2425],{"class":1052,"line":1814},[1050,2387,2388],{"class":1067},"    list_filter ",[1050,2390,1713],{"class":1091},[1050,2392,2393],{"class":1091}," [",[1050,2395,1115],{"class":1107},[1050,2397,2341],{"class":1111},[1050,2399,1115],{"class":1107},[1050,2401,1195],{"class":1091},[1050,2403,1121],{"class":1107},[1050,2405,1672],{"class":1111},[1050,2407,1115],{"class":1107},[1050,2409,1195],{"class":1091},[1050,2411,1121],{"class":1107},[1050,2413,2414],{"class":1111},"is_superuser",[1050,2416,1115],{"class":1107},[1050,2418,1195],{"class":1091},[1050,2420,1121],{"class":1107},[1050,2422,2363],{"class":1111},[1050,2424,1115],{"class":1107},[1050,2426,1275],{"class":1091},[1050,2428,2429],{"class":1052,"line":1832},[1050,2430,1598],{"class":1067},[1050,2432,2434,2437,2439,2442,2444,2447,2450],{"class":1052,"line":2433},41,[1050,2435,2436],{"class":1067},"    fieldsets ",[1050,2438,1713],{"class":1091},[1050,2440,2441],{"class":1067}," UserAdmin",[1050,2443,1532],{"class":1091},[1050,2445,2446],{"class":1649},"fieldsets",[1050,2448,2449],{"class":1658}," +",[1050,2451,1639],{"class":1091},[1050,2453,2455,2457,2459,2462,2464,2466],{"class":1052,"line":2454},42,[1050,2456,1953],{"class":1091},[1050,2458,1115],{"class":1107},[1050,2460,2461],{"class":1111},"Admin Security",[1050,2463,1115],{"class":1107},[1050,2465,1195],{"class":1091},[1050,2467,1175],{"class":1091},[1050,2469,2471,2473,2476,2478,2480],{"class":1052,"line":2470},43,[1050,2472,1181],{"class":1107},[1050,2474,2475],{"class":1111},"fields",[1050,2477,1115],{"class":1107},[1050,2479,1118],{"class":1091},[1050,2481,1639],{"class":1091},[1050,2483,2485,2488,2490,2492,2494],{"class":1052,"line":2484},44,[1050,2486,2487],{"class":1107},"                '",[1050,2489,2341],{"class":1111},[1050,2491,1115],{"class":1107},[1050,2493,1195],{"class":1091},[1050,2495,2356],{"class":1067},[1050,2497,2499,2501,2503,2505,2507],{"class":1052,"line":2498},45,[1050,2500,2487],{"class":1107},[1050,2502,2363],{"class":1111},[1050,2504,1115],{"class":1107},[1050,2506,1195],{"class":1091},[1050,2508,2356],{"class":1067},[1050,2510,2512,2514,2516,2518],{"class":1052,"line":2511},46,[1050,2513,2487],{"class":1107},[1050,2515,2372],{"class":1111},[1050,2517,1115],{"class":1107},[1050,2519,1129],{"class":1091},[1050,2521,2523,2525,2528],{"class":1052,"line":2522},47,[1050,2524,2487],{"class":1107},[1050,2526,2527],{"class":1111},"account_locked_until",[1050,2529,2375],{"class":1107},[1050,2531,2533],{"class":1052,"line":2532},48,[1050,2534,2535],{"class":1091},"            )\n",[1050,2537,2539],{"class":1052,"line":2538},49,[1050,2540,2541],{"class":1091},"        }),\n",[1050,2543,2545],{"class":1052,"line":2544},50,[1050,2546,2117],{"class":1091},[1050,2548,2550],{"class":1052,"line":2549},51,[1050,2551,1598],{"class":1067},[1050,2553,2555,2558,2560,2562,2564,2566,2568,2570,2572,2574,2576],{"class":1052,"line":2554},52,[1050,2556,2557],{"class":1067},"    readonly_fields ",[1050,2559,1713],{"class":1091},[1050,2561,2393],{"class":1091},[1050,2563,1115],{"class":1107},[1050,2565,2363],{"class":1111},[1050,2567,1115],{"class":1107},[1050,2569,1195],{"class":1091},[1050,2571,1121],{"class":1107},[1050,2573,2372],{"class":1111},[1050,2575,1115],{"class":1107},[1050,2577,1275],{"class":1091},[1050,2579,2581],{"class":1052,"line":2580},53,[1050,2582,1598],{"class":1067},[1050,2584,2586,2588,2591,2593,2595,2597,2599],{"class":1052,"line":2585},54,[1050,2587,1603],{"class":1565},[1050,2589,2590],{"class":1606}," get_queryset",[1050,2592,1573],{"class":1091},[1050,2594,1613],{"class":1612},[1050,2596,1195],{"class":1091},[1050,2598,1619],{"class":1618},[1050,2600,1580],{"class":1091},[1050,2602,2604,2606,2609],{"class":1052,"line":2603},55,[1050,2605,1626],{"class":1585},[1050,2607,2608],{"class":1589},"Filter users based on admin role",[1050,2610,1593],{"class":1585},[1050,2612,2614,2617,2619,2623,2626,2629,2631,2634],{"class":1052,"line":2613},56,[1050,2615,2616],{"class":1067},"        queryset ",[1050,2618,1713],{"class":1091},[1050,2620,2622],{"class":2621},"sa2tF"," super",[1050,2624,2625],{"class":1091},"().",[1050,2627,2628],{"class":1687},"get_queryset",[1050,2630,1573],{"class":1091},[1050,2632,2633],{"class":1687},"request",[1050,2635,1733],{"class":1091},[1050,2637,2639],{"class":1052,"line":2638},57,[1050,2640,2641],{"class":1067},"        \n",[1050,2643,2645,2648,2651,2653,2655,2657,2659,2661],{"class":1052,"line":2644},58,[1050,2646,2647],{"class":1063},"        if",[1050,2649,2650],{"class":1658}," not",[1050,2652,1619],{"class":1067},[1050,2654,1532],{"class":1091},[1050,2656,1650],{"class":1649},[1050,2658,1532],{"class":1091},[1050,2660,2414],{"class":1649},[1050,2662,2663],{"class":1091},":\n",[1050,2665,2667],{"class":1052,"line":2666},59,[1050,2668,2669],{"class":1056},"            # Non-superusers can only see users with lower privileges\n",[1050,2671,2673,2676,2678,2680,2682,2684,2686,2689,2691,2693,2695],{"class":1052,"line":2672},60,[1050,2674,2675],{"class":1063},"            if",[1050,2677,1619],{"class":1067},[1050,2679,1532],{"class":1091},[1050,2681,1650],{"class":1649},[1050,2683,1532],{"class":1091},[1050,2685,2341],{"class":1649},[1050,2687,2688],{"class":1658}," ==",[1050,2690,1121],{"class":1107},[1050,2692,1981],{"class":1111},[1050,2694,1115],{"class":1107},[1050,2696,2663],{"class":1091},[1050,2698,2700,2703,2705,2708,2710,2713],{"class":1052,"line":2699},61,[1050,2701,2702],{"class":1067},"                queryset ",[1050,2704,1713],{"class":1091},[1050,2706,2707],{"class":1067}," queryset",[1050,2709,1532],{"class":1091},[1050,2711,2712],{"class":1687},"filter",[1050,2714,2065],{"class":1091},[1050,2716,2718,2721,2724,2726,2728,2730,2732,2734,2736,2738],{"class":1052,"line":2717},62,[1050,2719,2720],{"class":1720},"                    admin_role__in",[1050,2722,2723],{"class":1091},"=[",[1050,2725,1115],{"class":1107},[1050,2727,1958],{"class":1111},[1050,2729,1115],{"class":1107},[1050,2731,1195],{"class":1091},[1050,2733,1121],{"class":1107},[1050,2735,2025],{"class":1111},[1050,2737,1115],{"class":1107},[1050,2739,1275],{"class":1091},[1050,2741,2743],{"class":1052,"line":2742},63,[1050,2744,2745],{"class":1091},"                )\n",[1050,2747,2749,2752,2754,2756,2758,2760,2762,2764,2766,2768,2770],{"class":1052,"line":2748},64,[1050,2750,2751],{"class":1063},"            elif",[1050,2753,1619],{"class":1067},[1050,2755,1532],{"class":1091},[1050,2757,1650],{"class":1649},[1050,2759,1532],{"class":1091},[1050,2761,2341],{"class":1649},[1050,2763,2688],{"class":1658},[1050,2765,1121],{"class":1107},[1050,2767,1958],{"class":1111},[1050,2769,1115],{"class":1107},[1050,2771,2663],{"class":1091},[1050,2773,2775,2777,2779,2781,2783,2785,2787,2789,2791,2793,2795,2797],{"class":1052,"line":2774},65,[1050,2776,2702],{"class":1067},[1050,2778,1713],{"class":1091},[1050,2780,2707],{"class":1067},[1050,2782,1532],{"class":1091},[1050,2784,2712],{"class":1687},[1050,2786,1573],{"class":1091},[1050,2788,2341],{"class":1720},[1050,2790,1713],{"class":1091},[1050,2792,1115],{"class":1107},[1050,2794,2025],{"class":1111},[1050,2796,1115],{"class":1107},[1050,2798,1733],{"class":1091},[1050,2800,2802],{"class":1052,"line":2801},66,[1050,2803,2641],{"class":1067},[1050,2805,2807,2809],{"class":1052,"line":2806},67,[1050,2808,1636],{"class":1063},[1050,2810,2811],{"class":1067}," queryset\n",[1050,2813,2815],{"class":1052,"line":2814},68,[1050,2816,1075],{"emptyLinePlaceholder":1074},[1050,2818,2820,2823,2825,2828,2830,2832,2834,2837,2839,2841],{"class":1052,"line":2819},69,[1050,2821,2822],{"class":1067},"admin",[1050,2824,1532],{"class":1091},[1050,2826,2827],{"class":1649},"site",[1050,2829,1532],{"class":1091},[1050,2831,1822],{"class":1687},[1050,2833,1573],{"class":1091},[1050,2835,2836],{"class":1687},"CustomUser",[1050,2838,1195],{"class":1091},[1050,2840,2287],{"class":1687},[1050,2842,1733],{"class":1091},[1030,2844,2846],{"id":2845},"url-and-network-security","URL and Network Security",[1035,2848,2850],{"id":2849},"secure-admin-urls","Secure Admin URLs",[1040,2852,2854],{"className":1042,"code":2853,"language":1044,"meta":1045,"style":1045},"# urls.py\nimport os\nfrom django.contrib import admin\nfrom django.urls import path, include\n\n# Use environment variable for admin URL\nADMIN_URL = os.environ.get('ADMIN_URL', 'admin/')\n\n# Ensure admin URL doesn't end with predictable patterns\nif ADMIN_URL in ['admin/', 'administrator/', 'manage/', 'control/']:\n    raise ValueError(\"Admin URL should not use common patterns\")\n\nurlpatterns = [\n    path(ADMIN_URL, admin.site.urls),\n    # ... other patterns\n]\n\n# Additional security: Random admin URL generation\nimport secrets\nimport string\n\ndef generate_admin_url():\n    \"\"\"Generate random admin URL\"\"\"\n    chars = string.ascii_letters + string.digits\n    return ''.join(secrets.choice(chars) for _ in range(16)) + '/'\n\n# Use in production deployment scripts\n",[1047,2855,2856,2861,2867,2881,2902,2906,2911,2950,2954,2959,3008,3028,3032,3041,3066,3071,3075,3079,3084,3091,3098,3102,3113,3122,3146,3207,3211],{"__ignoreMap":1045},[1050,2857,2858],{"class":1052,"line":1053},[1050,2859,2860],{"class":1056},"# urls.py\n",[1050,2862,2863,2865],{"class":1052,"line":1060},[1050,2864,1064],{"class":1063},[1050,2866,1068],{"class":1067},[1050,2868,2869,2871,2873,2875,2877,2879],{"class":1052,"line":1071},[1050,2870,1526],{"class":1063},[1050,2872,1749],{"class":1067},[1050,2874,1532],{"class":1091},[1050,2876,2232],{"class":1067},[1050,2878,1064],{"class":1063},[1050,2880,2237],{"class":1067},[1050,2882,2883,2885,2887,2889,2892,2894,2897,2899],{"class":1052,"line":1078},[1050,2884,1526],{"class":1063},[1050,2886,1749],{"class":1067},[1050,2888,1532],{"class":1091},[1050,2890,2891],{"class":1067},"urls ",[1050,2893,1064],{"class":1063},[1050,2895,2896],{"class":1067}," path",[1050,2898,1195],{"class":1091},[1050,2900,2901],{"class":1067}," include\n",[1050,2903,2904],{"class":1052,"line":1084},[1050,2905,1075],{"emptyLinePlaceholder":1074},[1050,2907,2908],{"class":1052,"line":1098},[1050,2909,2910],{"class":1056},"# Use environment variable for admin URL\n",[1050,2912,2913,2916,2918,2921,2923,2926,2928,2931,2933,2935,2937,2939,2941,2943,2946,2948],{"class":1052,"line":1104},[1050,2914,2915],{"class":1087},"ADMIN_URL",[1050,2917,1092],{"class":1091},[1050,2919,2920],{"class":1067}," os",[1050,2922,1532],{"class":1091},[1050,2924,2925],{"class":1649},"environ",[1050,2927,1532],{"class":1091},[1050,2929,2930],{"class":1687},"get",[1050,2932,1573],{"class":1091},[1050,2934,1115],{"class":1107},[1050,2936,2915],{"class":1111},[1050,2938,1115],{"class":1107},[1050,2940,1195],{"class":1091},[1050,2942,1121],{"class":1107},[1050,2944,2945],{"class":1111},"admin/",[1050,2947,1115],{"class":1107},[1050,2949,1733],{"class":1091},[1050,2951,2952],{"class":1052,"line":1132},[1050,2953,1075],{"emptyLinePlaceholder":1074},[1050,2955,2956],{"class":1052,"line":1138},[1050,2957,2958],{"class":1056},"# Ensure admin URL doesn't end with predictable patterns\n",[1050,2960,2961,2964,2967,2970,2972,2974,2976,2978,2980,2982,2985,2987,2989,2991,2994,2996,2998,3000,3003,3005],{"class":1052,"line":1143},[1050,2962,2963],{"class":1063},"if",[1050,2965,2966],{"class":1087}," ADMIN_URL",[1050,2968,2969],{"class":1658}," in",[1050,2971,2393],{"class":1091},[1050,2973,1115],{"class":1107},[1050,2975,2945],{"class":1111},[1050,2977,1115],{"class":1107},[1050,2979,1195],{"class":1091},[1050,2981,1121],{"class":1107},[1050,2983,2984],{"class":1111},"administrator/",[1050,2986,1115],{"class":1107},[1050,2988,1195],{"class":1091},[1050,2990,1121],{"class":1107},[1050,2992,2993],{"class":1111},"manage/",[1050,2995,1115],{"class":1107},[1050,2997,1195],{"class":1091},[1050,2999,1121],{"class":1107},[1050,3001,3002],{"class":1111},"control/",[1050,3004,1115],{"class":1107},[1050,3006,3007],{"class":1091},"]:\n",[1050,3009,3010,3013,3016,3018,3021,3024,3026],{"class":1052,"line":1163},[1050,3011,3012],{"class":1063},"    raise",[1050,3014,3015],{"class":2621}," ValueError",[1050,3017,1573],{"class":1091},[1050,3019,3020],{"class":1107},"\"",[1050,3022,3023],{"class":1111},"Admin URL should not use common patterns",[1050,3025,3020],{"class":1107},[1050,3027,1733],{"class":1091},[1050,3029,3030],{"class":1052,"line":1178},[1050,3031,1075],{"emptyLinePlaceholder":1074},[1050,3033,3034,3037,3039],{"class":1052,"line":1201},[1050,3035,3036],{"class":1067},"urlpatterns ",[1050,3038,1713],{"class":1091},[1050,3040,1095],{"class":1091},[1050,3042,3043,3046,3048,3050,3052,3055,3057,3059,3061,3064],{"class":1052,"line":1207},[1050,3044,3045],{"class":1687},"    path",[1050,3047,1573],{"class":1091},[1050,3049,2915],{"class":2087},[1050,3051,1195],{"class":1091},[1050,3053,3054],{"class":1687}," admin",[1050,3056,1532],{"class":1091},[1050,3058,2827],{"class":1649},[1050,3060,1532],{"class":1091},[1050,3062,3063],{"class":1649},"urls",[1050,3065,1972],{"class":1091},[1050,3067,3068],{"class":1052,"line":1212},[1050,3069,3070],{"class":1056},"    # ... other patterns\n",[1050,3072,3073],{"class":1052,"line":1217},[1050,3074,1275],{"class":1091},[1050,3076,3077],{"class":1052,"line":1237},[1050,3078,1075],{"emptyLinePlaceholder":1074},[1050,3080,3081],{"class":1052,"line":1242},[1050,3082,3083],{"class":1056},"# Additional security: Random admin URL generation\n",[1050,3085,3086,3088],{"class":1052,"line":1247},[1050,3087,1064],{"class":1063},[1050,3089,3090],{"class":1067}," secrets\n",[1050,3092,3093,3095],{"class":1052,"line":1267},[1050,3094,1064],{"class":1063},[1050,3096,3097],{"class":1067}," string\n",[1050,3099,3100],{"class":1052,"line":1272},[1050,3101,1075],{"emptyLinePlaceholder":1074},[1050,3103,3104,3107,3110],{"class":1052,"line":1278},[1050,3105,3106],{"class":1565},"def",[1050,3108,3109],{"class":1606}," generate_admin_url",[1050,3111,3112],{"class":1091},"():\n",[1050,3114,3115,3117,3120],{"class":1052,"line":1283},[1050,3116,1586],{"class":1585},[1050,3118,3119],{"class":1589},"Generate random admin URL",[1050,3121,1593],{"class":1585},[1050,3123,3124,3127,3129,3132,3134,3137,3139,3141,3143],{"class":1052,"line":1289},[1050,3125,3126],{"class":1067},"    chars ",[1050,3128,1713],{"class":1091},[1050,3130,3131],{"class":1067}," string",[1050,3133,1532],{"class":1091},[1050,3135,3136],{"class":1649},"ascii_letters",[1050,3138,2449],{"class":1658},[1050,3140,3131],{"class":1067},[1050,3142,1532],{"class":1091},[1050,3144,3145],{"class":1649},"digits\n",[1050,3147,3148,3151,3154,3156,3159,3161,3164,3166,3169,3171,3174,3177,3180,3183,3186,3190,3192,3195,3198,3200,3202,3205],{"class":1052,"line":1304},[1050,3149,3150],{"class":1063},"    return",[1050,3152,3153],{"class":1107}," ''",[1050,3155,1532],{"class":1091},[1050,3157,3158],{"class":1687},"join",[1050,3160,1573],{"class":1091},[1050,3162,3163],{"class":1687},"secrets",[1050,3165,1532],{"class":1091},[1050,3167,3168],{"class":1687},"choice",[1050,3170,1573],{"class":1091},[1050,3172,3173],{"class":1687},"chars",[1050,3175,3176],{"class":1091},")",[1050,3178,3179],{"class":1063}," for",[1050,3181,3182],{"class":1687}," _ ",[1050,3184,3185],{"class":1063},"in",[1050,3187,3189],{"class":3188},"sJdAF"," range",[1050,3191,1573],{"class":1091},[1050,3193,3194],{"class":1191},"16",[1050,3196,3197],{"class":1091},"))",[1050,3199,2449],{"class":1658},[1050,3201,1121],{"class":1107},[1050,3203,3204],{"class":1111},"/",[1050,3206,2375],{"class":1107},[1050,3208,3209],{"class":1052,"line":1317},[1050,3210,1075],{"emptyLinePlaceholder":1074},[1050,3212,3213],{"class":1052,"line":1331},[1050,3214,3215],{"class":1056},"# Use in production deployment scripts\n",[1035,3217,3219],{"id":3218},"ip-address-restrictions","IP Address Restrictions",[1040,3221,3223],{"className":1042,"code":3222,"language":1044,"meta":1045,"style":1045},"# middleware.py\nfrom django.http import HttpResponseForbidden\nfrom django.conf import settings\nimport ipaddress\nimport logging\n\nlogger = logging.getLogger(__name__)\n\nclass AdminIPRestrictionMiddleware:\n    \"\"\"Restrict admin access by IP address with logging\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n        self.allowed_ips = getattr(settings, 'ADMIN_ALLOWED_IPS', [])\n        self.admin_url = getattr(settings, 'ADMIN_URL', 'admin/')\n    \n    def __call__(self, request):\n        if request.path.startswith(f'/{self.admin_url}'):\n            client_ip = self.get_client_ip(request)\n            \n            if not self.is_ip_allowed(client_ip):\n                logger.warning(\n                    f'Admin access denied for IP {client_ip} - '\n                    f'User: {getattr(request.user, \"username\", \"Anonymous\")} - '\n                    f'Path: {request.path}'\n                )\n                return HttpResponseForbidden('Access denied from this IP address')\n        \n        return self.get_response(request)\n    \n    def get_client_ip(self, request):\n        \"\"\"Get real client IP address\"\"\"\n        # Check for IP in various headers (for load balancers/proxies)\n        ip_headers = [\n            'HTTP_X_FORWARDED_FOR',\n            'HTTP_X_REAL_IP',\n            'HTTP_CF_CONNECTING_IP',  # Cloudflare\n            'REMOTE_ADDR'\n        ]\n        \n        for header in ip_headers:\n            ip = request.META.get(header)\n            if ip:\n                # Handle comma-separated IPs (X-Forwarded-For)\n                ip = ip.split(',')[0].strip()\n                try:\n                    ipaddress.ip_address(ip)\n                    return ip\n                except ValueError:\n                    continue\n        \n        return request.META.get('REMOTE_ADDR', '0.0.0.0')\n    \n    def is_ip_allowed(self, ip):\n        \"\"\"Check if IP is in allowed networks\"\"\"\n        if not self.allowed_ips:\n            return True  # No restrictions if list is empty\n        \n        try:\n            client_ip = ipaddress.ip_address(ip)\n            for allowed_network in self.allowed_ips:\n                if client_ip in ipaddress.ip_network(allowed_network, strict=False):\n                    return True\n        except ValueError as e:\n            logger.error(f'Invalid IP address format: {ip} - {e}')\n        \n        return False\n\n# settings.py\nMIDDLEWARE = [\n    'myapp.middleware.AdminIPRestrictionMiddleware',\n    # ... other middleware\n]\n\n# Production IP restrictions\nADMIN_ALLOWED_IPS = [\n    '192.168.1.0/24',      # Office network\n    '10.0.0.0/8',          # VPN network\n    '203.0.113.100/32',    # Specific admin IP\n]\n",[1047,3224,3225,3230,3246,3262,3269,3276,3280,3303,3307,3316,3325,3329,3347,3362,3395,3430,3434,3451,3492,3513,3518,3538,3550,3567,3611,3630,3634,3653,3657,3673,3677,3694,3703,3708,3717,3728,3739,3753,3762,3767,3771,3786,3812,3821,3826,3861,3868,3885,3893,3902,3907,3911,3944,3948,3965,3974,3988,3998,4002,4009,4028,4046,4080,4086,4101,4138,4142,4149,4153,4157,4166,4178,4183,4188,4193,4199,4208,4223,4238,4253],{"__ignoreMap":1045},[1050,3226,3227],{"class":1052,"line":1053},[1050,3228,3229],{"class":1056},"# middleware.py\n",[1050,3231,3232,3234,3236,3238,3241,3243],{"class":1052,"line":1060},[1050,3233,1526],{"class":1063},[1050,3235,1749],{"class":1067},[1050,3237,1532],{"class":1091},[1050,3239,3240],{"class":1067},"http ",[1050,3242,1064],{"class":1063},[1050,3244,3245],{"class":1067}," HttpResponseForbidden\n",[1050,3247,3248,3250,3252,3254,3257,3259],{"class":1052,"line":1071},[1050,3249,1526],{"class":1063},[1050,3251,1749],{"class":1067},[1050,3253,1532],{"class":1091},[1050,3255,3256],{"class":1067},"conf ",[1050,3258,1064],{"class":1063},[1050,3260,3261],{"class":1067}," settings\n",[1050,3263,3264,3266],{"class":1052,"line":1078},[1050,3265,1064],{"class":1063},[1050,3267,3268],{"class":1067}," ipaddress\n",[1050,3270,3271,3273],{"class":1052,"line":1084},[1050,3272,1064],{"class":1063},[1050,3274,3275],{"class":1067}," logging\n",[1050,3277,3278],{"class":1052,"line":1098},[1050,3279,1075],{"emptyLinePlaceholder":1074},[1050,3281,3282,3285,3287,3290,3292,3295,3297,3301],{"class":1052,"line":1104},[1050,3283,3284],{"class":1067},"logger ",[1050,3286,1713],{"class":1091},[1050,3288,3289],{"class":1067}," logging",[1050,3291,1532],{"class":1091},[1050,3293,3294],{"class":1687},"getLogger",[1050,3296,1573],{"class":1091},[1050,3298,3300],{"class":3299},"s131V","__name__",[1050,3302,1733],{"class":1091},[1050,3304,3305],{"class":1052,"line":1132},[1050,3306,1075],{"emptyLinePlaceholder":1074},[1050,3308,3309,3311,3314],{"class":1052,"line":1138},[1050,3310,1566],{"class":1565},[1050,3312,3313],{"class":1569}," AdminIPRestrictionMiddleware",[1050,3315,2663],{"class":1091},[1050,3317,3318,3320,3323],{"class":1052,"line":1143},[1050,3319,1586],{"class":1585},[1050,3321,3322],{"class":1589},"Restrict admin access by IP address with logging",[1050,3324,1593],{"class":1585},[1050,3326,3327],{"class":1052,"line":1163},[1050,3328,1598],{"class":1067},[1050,3330,3331,3333,3336,3338,3340,3342,3345],{"class":1052,"line":1178},[1050,3332,1603],{"class":1565},[1050,3334,3335],{"class":3188}," __init__",[1050,3337,1573],{"class":1091},[1050,3339,1613],{"class":1612},[1050,3341,1195],{"class":1091},[1050,3343,3344],{"class":1618}," get_response",[1050,3346,1580],{"class":1091},[1050,3348,3349,3352,3354,3357,3359],{"class":1052,"line":1201},[1050,3350,3351],{"class":1087},"        self",[1050,3353,1532],{"class":1091},[1050,3355,3356],{"class":1649},"get_response",[1050,3358,1092],{"class":1091},[1050,3360,3361],{"class":1067}," get_response\n",[1050,3363,3364,3366,3368,3371,3373,3376,3378,3381,3383,3385,3388,3390,3392],{"class":1052,"line":1207},[1050,3365,3351],{"class":1087},[1050,3367,1532],{"class":1091},[1050,3369,3370],{"class":1649},"allowed_ips",[1050,3372,1092],{"class":1091},[1050,3374,3375],{"class":3188}," getattr",[1050,3377,1573],{"class":1091},[1050,3379,3380],{"class":1687},"settings",[1050,3382,1195],{"class":1091},[1050,3384,1121],{"class":1107},[1050,3386,3387],{"class":1111},"ADMIN_ALLOWED_IPS",[1050,3389,1115],{"class":1107},[1050,3391,1195],{"class":1091},[1050,3393,3394],{"class":1091}," [])\n",[1050,3396,3397,3399,3401,3404,3406,3408,3410,3412,3414,3416,3418,3420,3422,3424,3426,3428],{"class":1052,"line":1212},[1050,3398,3351],{"class":1087},[1050,3400,1532],{"class":1091},[1050,3402,3403],{"class":1649},"admin_url",[1050,3405,1092],{"class":1091},[1050,3407,3375],{"class":3188},[1050,3409,1573],{"class":1091},[1050,3411,3380],{"class":1687},[1050,3413,1195],{"class":1091},[1050,3415,1121],{"class":1107},[1050,3417,2915],{"class":1111},[1050,3419,1115],{"class":1107},[1050,3421,1195],{"class":1091},[1050,3423,1121],{"class":1107},[1050,3425,2945],{"class":1111},[1050,3427,1115],{"class":1107},[1050,3429,1733],{"class":1091},[1050,3431,3432],{"class":1052,"line":1217},[1050,3433,1598],{"class":1067},[1050,3435,3436,3438,3441,3443,3445,3447,3449],{"class":1052,"line":1237},[1050,3437,1603],{"class":1565},[1050,3439,3440],{"class":3188}," __call__",[1050,3442,1573],{"class":1091},[1050,3444,1613],{"class":1612},[1050,3446,1195],{"class":1091},[1050,3448,1619],{"class":1618},[1050,3450,1580],{"class":1091},[1050,3452,3453,3455,3457,3459,3462,3464,3467,3469,3472,3475,3479,3481,3483,3485,3488,3490],{"class":1052,"line":1242},[1050,3454,2647],{"class":1063},[1050,3456,1619],{"class":1067},[1050,3458,1532],{"class":1091},[1050,3460,3461],{"class":1649},"path",[1050,3463,1532],{"class":1091},[1050,3465,3466],{"class":1687},"startswith",[1050,3468,1573],{"class":1091},[1050,3470,3471],{"class":1565},"f",[1050,3473,3474],{"class":1111},"'/",[1050,3476,3478],{"class":3477},"s3h35","{",[1050,3480,1613],{"class":1087},[1050,3482,1532],{"class":1091},[1050,3484,3403],{"class":1649},[1050,3486,3487],{"class":3477},"}",[1050,3489,1115],{"class":1111},[1050,3491,1580],{"class":1091},[1050,3493,3494,3497,3499,3502,3504,3507,3509,3511],{"class":1052,"line":1247},[1050,3495,3496],{"class":1067},"            client_ip ",[1050,3498,1713],{"class":1091},[1050,3500,3501],{"class":1087}," self",[1050,3503,1532],{"class":1091},[1050,3505,3506],{"class":1687},"get_client_ip",[1050,3508,1573],{"class":1091},[1050,3510,2633],{"class":1687},[1050,3512,1733],{"class":1091},[1050,3514,3515],{"class":1052,"line":1267},[1050,3516,3517],{"class":1067},"            \n",[1050,3519,3520,3522,3524,3526,3528,3531,3533,3536],{"class":1052,"line":1272},[1050,3521,2675],{"class":1063},[1050,3523,2650],{"class":1658},[1050,3525,3501],{"class":1087},[1050,3527,1532],{"class":1091},[1050,3529,3530],{"class":1687},"is_ip_allowed",[1050,3532,1573],{"class":1091},[1050,3534,3535],{"class":1687},"client_ip",[1050,3537,1580],{"class":1091},[1050,3539,3540,3543,3545,3548],{"class":1052,"line":1278},[1050,3541,3542],{"class":1067},"                logger",[1050,3544,1532],{"class":1091},[1050,3546,3547],{"class":1687},"warning",[1050,3549,2065],{"class":1091},[1050,3551,3552,3555,3558,3560,3562,3564],{"class":1052,"line":1283},[1050,3553,3554],{"class":1565},"                    f",[1050,3556,3557],{"class":1111},"'Admin access denied for IP ",[1050,3559,3478],{"class":3477},[1050,3561,3535],{"class":1687},[1050,3563,3487],{"class":3477},[1050,3565,3566],{"class":1111}," - '\n",[1050,3568,3569,3571,3574,3576,3579,3581,3583,3585,3587,3589,3592,3594,3596,3598,3600,3603,3605,3607,3609],{"class":1052,"line":1289},[1050,3570,3554],{"class":1565},[1050,3572,3573],{"class":1111},"'User: ",[1050,3575,3478],{"class":3477},[1050,3577,3578],{"class":3188},"getattr",[1050,3580,1573],{"class":1091},[1050,3582,2633],{"class":1687},[1050,3584,1532],{"class":1091},[1050,3586,1650],{"class":1649},[1050,3588,1195],{"class":1091},[1050,3590,3591],{"class":1107}," \"",[1050,3593,2323],{"class":1111},[1050,3595,3020],{"class":1107},[1050,3597,1195],{"class":1091},[1050,3599,3591],{"class":1107},[1050,3601,3602],{"class":1111},"Anonymous",[1050,3604,3020],{"class":1107},[1050,3606,3176],{"class":1091},[1050,3608,3487],{"class":3477},[1050,3610,3566],{"class":1111},[1050,3612,3613,3615,3618,3620,3622,3624,3626,3628],{"class":1052,"line":1304},[1050,3614,3554],{"class":1565},[1050,3616,3617],{"class":1111},"'Path: ",[1050,3619,3478],{"class":3477},[1050,3621,2633],{"class":1687},[1050,3623,1532],{"class":1091},[1050,3625,3461],{"class":1649},[1050,3627,3487],{"class":3477},[1050,3629,2375],{"class":1111},[1050,3631,3632],{"class":1052,"line":1317},[1050,3633,2745],{"class":1091},[1050,3635,3636,3639,3642,3644,3646,3649,3651],{"class":1052,"line":1331},[1050,3637,3638],{"class":1063},"                return",[1050,3640,3641],{"class":1687}," HttpResponseForbidden",[1050,3643,1573],{"class":1091},[1050,3645,1115],{"class":1107},[1050,3647,3648],{"class":1111},"Access denied from this IP address",[1050,3650,1115],{"class":1107},[1050,3652,1733],{"class":1091},[1050,3654,3655],{"class":1052,"line":1342},[1050,3656,2641],{"class":1067},[1050,3658,3659,3661,3663,3665,3667,3669,3671],{"class":1052,"line":1347},[1050,3660,1636],{"class":1063},[1050,3662,3501],{"class":1087},[1050,3664,1532],{"class":1091},[1050,3666,3356],{"class":1687},[1050,3668,1573],{"class":1091},[1050,3670,2633],{"class":1687},[1050,3672,1733],{"class":1091},[1050,3674,3675],{"class":1052,"line":1353},[1050,3676,1598],{"class":1067},[1050,3678,3679,3681,3684,3686,3688,3690,3692],{"class":1052,"line":1363},[1050,3680,1603],{"class":1565},[1050,3682,3683],{"class":1606}," get_client_ip",[1050,3685,1573],{"class":1091},[1050,3687,1613],{"class":1612},[1050,3689,1195],{"class":1091},[1050,3691,1619],{"class":1618},[1050,3693,1580],{"class":1091},[1050,3695,3696,3698,3701],{"class":1052,"line":1373},[1050,3697,1626],{"class":1585},[1050,3699,3700],{"class":1589},"Get real client IP address",[1050,3702,1593],{"class":1585},[1050,3704,3705],{"class":1052,"line":1383},[1050,3706,3707],{"class":1056},"        # Check for IP in various headers (for load balancers/proxies)\n",[1050,3709,3710,3713,3715],{"class":1052,"line":1388},[1050,3711,3712],{"class":1067},"        ip_headers ",[1050,3714,1713],{"class":1091},[1050,3716,1095],{"class":1091},[1050,3718,3719,3721,3724,3726],{"class":1052,"line":1394},[1050,3720,1181],{"class":1107},[1050,3722,3723],{"class":1111},"HTTP_X_FORWARDED_FOR",[1050,3725,1115],{"class":1107},[1050,3727,1129],{"class":1091},[1050,3729,3730,3732,3735,3737],{"class":1052,"line":1777},[1050,3731,1181],{"class":1107},[1050,3733,3734],{"class":1111},"HTTP_X_REAL_IP",[1050,3736,1115],{"class":1107},[1050,3738,1129],{"class":1091},[1050,3740,3741,3743,3746,3748,3750],{"class":1052,"line":1793},[1050,3742,1181],{"class":1107},[1050,3744,3745],{"class":1111},"HTTP_CF_CONNECTING_IP",[1050,3747,1115],{"class":1107},[1050,3749,1195],{"class":1091},[1050,3751,3752],{"class":1056},"  # Cloudflare\n",[1050,3754,3755,3757,3760],{"class":1052,"line":1809},[1050,3756,1181],{"class":1107},[1050,3758,3759],{"class":1111},"REMOTE_ADDR",[1050,3761,2375],{"class":1107},[1050,3763,3764],{"class":1052,"line":1814},[1050,3765,3766],{"class":1091},"        ]\n",[1050,3768,3769],{"class":1052,"line":1832},[1050,3770,2641],{"class":1067},[1050,3772,3773,3776,3779,3781,3784],{"class":1052,"line":2433},[1050,3774,3775],{"class":1063},"        for",[1050,3777,3778],{"class":1067}," header ",[1050,3780,3185],{"class":1063},[1050,3782,3783],{"class":1067}," ip_headers",[1050,3785,2663],{"class":1091},[1050,3787,3788,3791,3793,3795,3797,3801,3803,3805,3807,3810],{"class":1052,"line":2454},[1050,3789,3790],{"class":1067},"            ip ",[1050,3792,1713],{"class":1091},[1050,3794,1619],{"class":1067},[1050,3796,1532],{"class":1091},[1050,3798,3800],{"class":3799},"sFGJz","META",[1050,3802,1532],{"class":1091},[1050,3804,2930],{"class":1687},[1050,3806,1573],{"class":1091},[1050,3808,3809],{"class":1687},"header",[1050,3811,1733],{"class":1091},[1050,3813,3814,3816,3819],{"class":1052,"line":2470},[1050,3815,2675],{"class":1063},[1050,3817,3818],{"class":1067}," ip",[1050,3820,2663],{"class":1091},[1050,3822,3823],{"class":1052,"line":2484},[1050,3824,3825],{"class":1056},"                # Handle comma-separated IPs (X-Forwarded-For)\n",[1050,3827,3828,3831,3833,3835,3837,3840,3842,3844,3846,3848,3851,3853,3856,3859],{"class":1052,"line":2498},[1050,3829,3830],{"class":1067},"                ip ",[1050,3832,1713],{"class":1091},[1050,3834,3818],{"class":1067},[1050,3836,1532],{"class":1091},[1050,3838,3839],{"class":1687},"split",[1050,3841,1573],{"class":1091},[1050,3843,1115],{"class":1107},[1050,3845,1195],{"class":1111},[1050,3847,1115],{"class":1107},[1050,3849,3850],{"class":1091},")[",[1050,3852,2179],{"class":1191},[1050,3854,3855],{"class":1091},"].",[1050,3857,3858],{"class":1687},"strip",[1050,3860,1691],{"class":1091},[1050,3862,3863,3866],{"class":1052,"line":2511},[1050,3864,3865],{"class":1063},"                try",[1050,3867,2663],{"class":1091},[1050,3869,3870,3873,3875,3878,3880,3883],{"class":1052,"line":2522},[1050,3871,3872],{"class":1067},"                    ipaddress",[1050,3874,1532],{"class":1091},[1050,3876,3877],{"class":1687},"ip_address",[1050,3879,1573],{"class":1091},[1050,3881,3882],{"class":1687},"ip",[1050,3884,1733],{"class":1091},[1050,3886,3887,3890],{"class":1052,"line":2532},[1050,3888,3889],{"class":1063},"                    return",[1050,3891,3892],{"class":1067}," ip\n",[1050,3894,3895,3898,3900],{"class":1052,"line":2538},[1050,3896,3897],{"class":1063},"                except",[1050,3899,3015],{"class":2621},[1050,3901,2663],{"class":1091},[1050,3903,3904],{"class":1052,"line":2544},[1050,3905,3906],{"class":1063},"                    continue\n",[1050,3908,3909],{"class":1052,"line":2549},[1050,3910,2641],{"class":1067},[1050,3912,3913,3915,3917,3919,3921,3923,3925,3927,3929,3931,3933,3935,3937,3940,3942],{"class":1052,"line":2554},[1050,3914,1636],{"class":1063},[1050,3916,1619],{"class":1067},[1050,3918,1532],{"class":1091},[1050,3920,3800],{"class":3799},[1050,3922,1532],{"class":1091},[1050,3924,2930],{"class":1687},[1050,3926,1573],{"class":1091},[1050,3928,1115],{"class":1107},[1050,3930,3759],{"class":1111},[1050,3932,1115],{"class":1107},[1050,3934,1195],{"class":1091},[1050,3936,1121],{"class":1107},[1050,3938,3939],{"class":1111},"0.0.0.0",[1050,3941,1115],{"class":1107},[1050,3943,1733],{"class":1091},[1050,3945,3946],{"class":1052,"line":2580},[1050,3947,1598],{"class":1067},[1050,3949,3950,3952,3955,3957,3959,3961,3963],{"class":1052,"line":2585},[1050,3951,1603],{"class":1565},[1050,3953,3954],{"class":1606}," is_ip_allowed",[1050,3956,1573],{"class":1091},[1050,3958,1613],{"class":1612},[1050,3960,1195],{"class":1091},[1050,3962,3818],{"class":1618},[1050,3964,1580],{"class":1091},[1050,3966,3967,3969,3972],{"class":1052,"line":2603},[1050,3968,1626],{"class":1585},[1050,3970,3971],{"class":1589},"Check if IP is in allowed networks",[1050,3973,1593],{"class":1585},[1050,3975,3976,3978,3980,3982,3984,3986],{"class":1052,"line":2613},[1050,3977,2647],{"class":1063},[1050,3979,2650],{"class":1658},[1050,3981,3501],{"class":1087},[1050,3983,1532],{"class":1091},[1050,3985,3370],{"class":1649},[1050,3987,2663],{"class":1091},[1050,3989,3990,3993,3995],{"class":1052,"line":2638},[1050,3991,3992],{"class":1063},"            return",[1050,3994,1298],{"class":1297},[1050,3996,3997],{"class":1056},"  # No restrictions if list is empty\n",[1050,3999,4000],{"class":1052,"line":2644},[1050,4001,2641],{"class":1067},[1050,4003,4004,4007],{"class":1052,"line":2666},[1050,4005,4006],{"class":1063},"        try",[1050,4008,2663],{"class":1091},[1050,4010,4011,4013,4015,4018,4020,4022,4024,4026],{"class":1052,"line":2672},[1050,4012,3496],{"class":1067},[1050,4014,1713],{"class":1091},[1050,4016,4017],{"class":1067}," ipaddress",[1050,4019,1532],{"class":1091},[1050,4021,3877],{"class":1687},[1050,4023,1573],{"class":1091},[1050,4025,3882],{"class":1687},[1050,4027,1733],{"class":1091},[1050,4029,4030,4033,4036,4038,4040,4042,4044],{"class":1052,"line":2699},[1050,4031,4032],{"class":1063},"            for",[1050,4034,4035],{"class":1067}," allowed_network ",[1050,4037,3185],{"class":1063},[1050,4039,3501],{"class":1087},[1050,4041,1532],{"class":1091},[1050,4043,3370],{"class":1649},[1050,4045,2663],{"class":1091},[1050,4047,4048,4051,4054,4056,4058,4060,4063,4065,4068,4070,4073,4075,4078],{"class":1052,"line":2717},[1050,4049,4050],{"class":1063},"                if",[1050,4052,4053],{"class":1067}," client_ip ",[1050,4055,3185],{"class":1658},[1050,4057,4017],{"class":1067},[1050,4059,1532],{"class":1091},[1050,4061,4062],{"class":1687},"ip_network",[1050,4064,1573],{"class":1091},[1050,4066,4067],{"class":1687},"allowed_network",[1050,4069,1195],{"class":1091},[1050,4071,4072],{"class":1720}," strict",[1050,4074,1713],{"class":1091},[1050,4076,4077],{"class":1297},"False",[1050,4079,1580],{"class":1091},[1050,4081,4082,4084],{"class":1052,"line":2742},[1050,4083,3889],{"class":1063},[1050,4085,1339],{"class":1297},[1050,4087,4088,4091,4093,4096,4099],{"class":1052,"line":2748},[1050,4089,4090],{"class":1063},"        except",[1050,4092,3015],{"class":2621},[1050,4094,4095],{"class":1063}," as",[1050,4097,4098],{"class":1067}," e",[1050,4100,2663],{"class":1091},[1050,4102,4103,4106,4108,4111,4113,4115,4118,4120,4122,4124,4127,4129,4132,4134,4136],{"class":1052,"line":2774},[1050,4104,4105],{"class":1067},"            logger",[1050,4107,1532],{"class":1091},[1050,4109,4110],{"class":1687},"error",[1050,4112,1573],{"class":1091},[1050,4114,3471],{"class":1565},[1050,4116,4117],{"class":1111},"'Invalid IP address format: ",[1050,4119,3478],{"class":3477},[1050,4121,3882],{"class":1687},[1050,4123,3487],{"class":3477},[1050,4125,4126],{"class":1111}," - ",[1050,4128,3478],{"class":3477},[1050,4130,4131],{"class":1687},"e",[1050,4133,3487],{"class":3477},[1050,4135,1115],{"class":1111},[1050,4137,1733],{"class":1091},[1050,4139,4140],{"class":1052,"line":2801},[1050,4141,2641],{"class":1067},[1050,4143,4144,4146],{"class":1052,"line":2806},[1050,4145,1636],{"class":1063},[1050,4147,4148],{"class":1297}," False\n",[1050,4150,4151],{"class":1052,"line":2814},[1050,4152,1075],{"emptyLinePlaceholder":1074},[1050,4154,4155],{"class":1052,"line":2819},[1050,4156,1057],{"class":1056},[1050,4158,4160,4162,4164],{"class":1052,"line":4159},70,[1050,4161,1488],{"class":1087},[1050,4163,1092],{"class":1091},[1050,4165,1095],{"class":1091},[1050,4167,4169,4171,4174,4176],{"class":1052,"line":4168},71,[1050,4170,1441],{"class":1107},[1050,4172,4173],{"class":1111},"myapp.middleware.AdminIPRestrictionMiddleware",[1050,4175,1115],{"class":1107},[1050,4177,1129],{"class":1091},[1050,4179,4181],{"class":1052,"line":4180},72,[1050,4182,1508],{"class":1056},[1050,4184,4186],{"class":1052,"line":4185},73,[1050,4187,1275],{"class":1091},[1050,4189,4191],{"class":1052,"line":4190},74,[1050,4192,1075],{"emptyLinePlaceholder":1074},[1050,4194,4196],{"class":1052,"line":4195},75,[1050,4197,4198],{"class":1056},"# Production IP restrictions\n",[1050,4200,4202,4204,4206],{"class":1052,"line":4201},76,[1050,4203,3387],{"class":1087},[1050,4205,1092],{"class":1091},[1050,4207,1095],{"class":1091},[1050,4209,4211,4213,4216,4218,4220],{"class":1052,"line":4210},77,[1050,4212,1441],{"class":1107},[1050,4214,4215],{"class":1111},"192.168.1.0/24",[1050,4217,1115],{"class":1107},[1050,4219,1195],{"class":1091},[1050,4221,4222],{"class":1056},"      # Office network\n",[1050,4224,4226,4228,4231,4233,4235],{"class":1052,"line":4225},78,[1050,4227,1441],{"class":1107},[1050,4229,4230],{"class":1111},"10.0.0.0/8",[1050,4232,1115],{"class":1107},[1050,4234,1195],{"class":1091},[1050,4236,4237],{"class":1056},"          # VPN network\n",[1050,4239,4241,4243,4246,4248,4250],{"class":1052,"line":4240},79,[1050,4242,1441],{"class":1107},[1050,4244,4245],{"class":1111},"203.0.113.100/32",[1050,4247,1115],{"class":1107},[1050,4249,1195],{"class":1091},[1050,4251,4252],{"class":1056},"    # Specific admin IP\n",[1050,4254,4256],{"class":1052,"line":4255},80,[1050,4257,1275],{"class":1091},[1035,4259,4261],{"id":4260},"rate-limiting-and-brute-force-protection","Rate Limiting and Brute Force Protection",[1040,4263,4265],{"className":1042,"code":4264,"language":1044,"meta":1045,"style":1045},"# middleware.py\nfrom django.core.cache import cache\nfrom django.http import HttpResponseTooManyRequests\nfrom django.contrib.auth import authenticate\nimport time\n\nclass AdminRateLimitMiddleware:\n    \"\"\"Rate limiting for admin login attempts\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n        self.max_attempts = 5\n        self.lockout_duration = 900  # 15 minutes\n        self.admin_url = getattr(settings, 'ADMIN_URL', 'admin/')\n    \n    def __call__(self, request):\n        if self.is_admin_login(request):\n            client_ip = self.get_client_ip(request)\n            \n            if self.is_rate_limited(client_ip):\n                return HttpResponseTooManyRequests(\n                    'Too many login attempts. Please try again later.'\n                )\n            \n            # Track failed attempts\n            if request.method == 'POST':\n                username = request.POST.get('username')\n                password = request.POST.get('password')\n                \n                if username and password:\n                    user = authenticate(username=username, password=password)\n                    if not user:\n                        self.record_failed_attempt(client_ip, username)\n        \n        return self.get_response(request)\n    \n    def is_admin_login(self, request):\n        \"\"\"Check if this is an admin login request\"\"\"\n        return (\n            request.path.startswith(f'/{self.admin_url}login/') or\n            request.path == f'/{self.admin_url}login'\n        )\n    \n    def is_rate_limited(self, ip):\n        \"\"\"Check if IP is rate limited\"\"\"\n        cache_key = f'admin_login_attempts:{ip}'\n        attempts = cache.get(cache_key, 0)\n        return attempts >= self.max_attempts\n    \n    def record_failed_attempt(self, ip, username):\n        \"\"\"Record failed login attempt\"\"\"\n        cache_key = f'admin_login_attempts:{ip}'\n        attempts = cache.get(cache_key, 0) + 1\n        cache.set(cache_key, attempts, self.lockout_duration)\n        \n        # Log security event\n        logger.warning(\n            f'Failed admin login attempt #{attempts} - '\n            f'IP: {ip}, Username: {username}'\n        )\n        \n        # Lock user account after multiple failures\n        if attempts >= 3:\n            try:\n                from django.contrib.auth.models import User\n                user = User.objects.get(username=username)\n                user.failed_login_attempts = attempts\n                if attempts >= self.max_attempts:\n                    user.account_locked_until = timezone.now() + timedelta(\n                        seconds=self.lockout_duration\n                    )\n                user.save()\n            except User.DoesNotExist:\n                pass\n",[1047,4266,4267,4271,4292,4307,4327,4334,4338,4347,4356,4360,4376,4388,4402,4419,4453,4457,4473,4490,4508,4512,4529,4538,4548,4552,4556,4561,4583,4610,4638,4643,4658,4686,4698,4719,4723,4739,4743,4760,4769,4775,4811,4839,4843,4847,4864,4873,4893,4919,4936,4940,4961,4970,4988,5015,5044,5048,5053,5064,5081,5105,5109,5113,5118,5131,5138,5162,5190,5204,5220,5249,5263,5268,5279,5293],{"__ignoreMap":1045},[1050,4268,4269],{"class":1052,"line":1053},[1050,4270,3229],{"class":1056},[1050,4272,4273,4275,4277,4279,4282,4284,4287,4289],{"class":1052,"line":1060},[1050,4274,1526],{"class":1063},[1050,4276,1749],{"class":1067},[1050,4278,1532],{"class":1091},[1050,4280,4281],{"class":1067},"core",[1050,4283,1532],{"class":1091},[1050,4285,4286],{"class":1067},"cache ",[1050,4288,1064],{"class":1063},[1050,4290,4291],{"class":1067}," cache\n",[1050,4293,4294,4296,4298,4300,4302,4304],{"class":1052,"line":1071},[1050,4295,1526],{"class":1063},[1050,4297,1749],{"class":1067},[1050,4299,1532],{"class":1091},[1050,4301,3240],{"class":1067},[1050,4303,1064],{"class":1063},[1050,4305,4306],{"class":1067}," HttpResponseTooManyRequests\n",[1050,4308,4309,4311,4313,4315,4317,4319,4322,4324],{"class":1052,"line":1078},[1050,4310,1526],{"class":1063},[1050,4312,1749],{"class":1067},[1050,4314,1532],{"class":1091},[1050,4316,1754],{"class":1067},[1050,4318,1532],{"class":1091},[1050,4320,4321],{"class":1067},"auth ",[1050,4323,1064],{"class":1063},[1050,4325,4326],{"class":1067}," authenticate\n",[1050,4328,4329,4331],{"class":1052,"line":1084},[1050,4330,1064],{"class":1063},[1050,4332,4333],{"class":1067}," time\n",[1050,4335,4336],{"class":1052,"line":1098},[1050,4337,1075],{"emptyLinePlaceholder":1074},[1050,4339,4340,4342,4345],{"class":1052,"line":1104},[1050,4341,1566],{"class":1565},[1050,4343,4344],{"class":1569}," AdminRateLimitMiddleware",[1050,4346,2663],{"class":1091},[1050,4348,4349,4351,4354],{"class":1052,"line":1132},[1050,4350,1586],{"class":1585},[1050,4352,4353],{"class":1589},"Rate limiting for admin login attempts",[1050,4355,1593],{"class":1585},[1050,4357,4358],{"class":1052,"line":1138},[1050,4359,1598],{"class":1067},[1050,4361,4362,4364,4366,4368,4370,4372,4374],{"class":1052,"line":1143},[1050,4363,1603],{"class":1565},[1050,4365,3335],{"class":3188},[1050,4367,1573],{"class":1091},[1050,4369,1613],{"class":1612},[1050,4371,1195],{"class":1091},[1050,4373,3344],{"class":1618},[1050,4375,1580],{"class":1091},[1050,4377,4378,4380,4382,4384,4386],{"class":1052,"line":1163},[1050,4379,3351],{"class":1087},[1050,4381,1532],{"class":1091},[1050,4383,3356],{"class":1649},[1050,4385,1092],{"class":1091},[1050,4387,3361],{"class":1067},[1050,4389,4390,4392,4394,4397,4399],{"class":1052,"line":1178},[1050,4391,3351],{"class":1087},[1050,4393,1532],{"class":1091},[1050,4395,4396],{"class":1649},"max_attempts",[1050,4398,1092],{"class":1091},[1050,4400,4401],{"class":1191}," 5\n",[1050,4403,4404,4406,4408,4411,4413,4416],{"class":1052,"line":1201},[1050,4405,3351],{"class":1087},[1050,4407,1532],{"class":1091},[1050,4409,4410],{"class":1649},"lockout_duration",[1050,4412,1092],{"class":1091},[1050,4414,4415],{"class":1191}," 900",[1050,4417,4418],{"class":1056},"  # 15 minutes\n",[1050,4420,4421,4423,4425,4427,4429,4431,4433,4435,4437,4439,4441,4443,4445,4447,4449,4451],{"class":1052,"line":1207},[1050,4422,3351],{"class":1087},[1050,4424,1532],{"class":1091},[1050,4426,3403],{"class":1649},[1050,4428,1092],{"class":1091},[1050,4430,3375],{"class":3188},[1050,4432,1573],{"class":1091},[1050,4434,3380],{"class":1687},[1050,4436,1195],{"class":1091},[1050,4438,1121],{"class":1107},[1050,4440,2915],{"class":1111},[1050,4442,1115],{"class":1107},[1050,4444,1195],{"class":1091},[1050,4446,1121],{"class":1107},[1050,4448,2945],{"class":1111},[1050,4450,1115],{"class":1107},[1050,4452,1733],{"class":1091},[1050,4454,4455],{"class":1052,"line":1212},[1050,4456,1598],{"class":1067},[1050,4458,4459,4461,4463,4465,4467,4469,4471],{"class":1052,"line":1217},[1050,4460,1603],{"class":1565},[1050,4462,3440],{"class":3188},[1050,4464,1573],{"class":1091},[1050,4466,1613],{"class":1612},[1050,4468,1195],{"class":1091},[1050,4470,1619],{"class":1618},[1050,4472,1580],{"class":1091},[1050,4474,4475,4477,4479,4481,4484,4486,4488],{"class":1052,"line":1237},[1050,4476,2647],{"class":1063},[1050,4478,3501],{"class":1087},[1050,4480,1532],{"class":1091},[1050,4482,4483],{"class":1687},"is_admin_login",[1050,4485,1573],{"class":1091},[1050,4487,2633],{"class":1687},[1050,4489,1580],{"class":1091},[1050,4491,4492,4494,4496,4498,4500,4502,4504,4506],{"class":1052,"line":1242},[1050,4493,3496],{"class":1067},[1050,4495,1713],{"class":1091},[1050,4497,3501],{"class":1087},[1050,4499,1532],{"class":1091},[1050,4501,3506],{"class":1687},[1050,4503,1573],{"class":1091},[1050,4505,2633],{"class":1687},[1050,4507,1733],{"class":1091},[1050,4509,4510],{"class":1052,"line":1247},[1050,4511,3517],{"class":1067},[1050,4513,4514,4516,4518,4520,4523,4525,4527],{"class":1052,"line":1267},[1050,4515,2675],{"class":1063},[1050,4517,3501],{"class":1087},[1050,4519,1532],{"class":1091},[1050,4521,4522],{"class":1687},"is_rate_limited",[1050,4524,1573],{"class":1091},[1050,4526,3535],{"class":1687},[1050,4528,1580],{"class":1091},[1050,4530,4531,4533,4536],{"class":1052,"line":1272},[1050,4532,3638],{"class":1063},[1050,4534,4535],{"class":1687}," HttpResponseTooManyRequests",[1050,4537,2065],{"class":1091},[1050,4539,4540,4543,4546],{"class":1052,"line":1278},[1050,4541,4542],{"class":1107},"                    '",[1050,4544,4545],{"class":1111},"Too many login attempts. Please try again later.",[1050,4547,2375],{"class":1107},[1050,4549,4550],{"class":1052,"line":1283},[1050,4551,2745],{"class":1091},[1050,4553,4554],{"class":1052,"line":1289},[1050,4555,3517],{"class":1067},[1050,4557,4558],{"class":1052,"line":1304},[1050,4559,4560],{"class":1056},"            # Track failed attempts\n",[1050,4562,4563,4565,4567,4569,4572,4574,4576,4579,4581],{"class":1052,"line":1317},[1050,4564,2675],{"class":1063},[1050,4566,1619],{"class":1067},[1050,4568,1532],{"class":1091},[1050,4570,4571],{"class":1649},"method",[1050,4573,2688],{"class":1658},[1050,4575,1121],{"class":1107},[1050,4577,4578],{"class":1111},"POST",[1050,4580,1115],{"class":1107},[1050,4582,2663],{"class":1091},[1050,4584,4585,4588,4590,4592,4594,4596,4598,4600,4602,4604,4606,4608],{"class":1052,"line":1331},[1050,4586,4587],{"class":1067},"                username ",[1050,4589,1713],{"class":1091},[1050,4591,1619],{"class":1067},[1050,4593,1532],{"class":1091},[1050,4595,4578],{"class":3799},[1050,4597,1532],{"class":1091},[1050,4599,2930],{"class":1687},[1050,4601,1573],{"class":1091},[1050,4603,1115],{"class":1107},[1050,4605,2323],{"class":1111},[1050,4607,1115],{"class":1107},[1050,4609,1733],{"class":1091},[1050,4611,4612,4615,4617,4619,4621,4623,4625,4627,4629,4631,4634,4636],{"class":1052,"line":1342},[1050,4613,4614],{"class":1067},"                password ",[1050,4616,1713],{"class":1091},[1050,4618,1619],{"class":1067},[1050,4620,1532],{"class":1091},[1050,4622,4578],{"class":3799},[1050,4624,1532],{"class":1091},[1050,4626,2930],{"class":1687},[1050,4628,1573],{"class":1091},[1050,4630,1115],{"class":1107},[1050,4632,4633],{"class":1111},"password",[1050,4635,1115],{"class":1107},[1050,4637,1733],{"class":1091},[1050,4639,4640],{"class":1052,"line":1347},[1050,4641,4642],{"class":1067},"                \n",[1050,4644,4645,4647,4650,4653,4656],{"class":1052,"line":1353},[1050,4646,4050],{"class":1063},[1050,4648,4649],{"class":1067}," username ",[1050,4651,4652],{"class":1658},"and",[1050,4654,4655],{"class":1067}," password",[1050,4657,2663],{"class":1091},[1050,4659,4660,4663,4665,4668,4670,4672,4674,4676,4678,4680,4682,4684],{"class":1052,"line":1363},[1050,4661,4662],{"class":1067},"                    user ",[1050,4664,1713],{"class":1091},[1050,4666,4667],{"class":1687}," authenticate",[1050,4669,1573],{"class":1091},[1050,4671,2323],{"class":1720},[1050,4673,1713],{"class":1091},[1050,4675,2323],{"class":1687},[1050,4677,1195],{"class":1091},[1050,4679,4655],{"class":1720},[1050,4681,1713],{"class":1091},[1050,4683,4633],{"class":1687},[1050,4685,1733],{"class":1091},[1050,4687,4688,4691,4693,4696],{"class":1052,"line":1373},[1050,4689,4690],{"class":1063},"                    if",[1050,4692,2650],{"class":1658},[1050,4694,4695],{"class":1067}," user",[1050,4697,2663],{"class":1091},[1050,4699,4700,4703,4705,4708,4710,4712,4714,4717],{"class":1052,"line":1383},[1050,4701,4702],{"class":1087},"                        self",[1050,4704,1532],{"class":1091},[1050,4706,4707],{"class":1687},"record_failed_attempt",[1050,4709,1573],{"class":1091},[1050,4711,3535],{"class":1687},[1050,4713,1195],{"class":1091},[1050,4715,4716],{"class":1687}," username",[1050,4718,1733],{"class":1091},[1050,4720,4721],{"class":1052,"line":1388},[1050,4722,2641],{"class":1067},[1050,4724,4725,4727,4729,4731,4733,4735,4737],{"class":1052,"line":1394},[1050,4726,1636],{"class":1063},[1050,4728,3501],{"class":1087},[1050,4730,1532],{"class":1091},[1050,4732,3356],{"class":1687},[1050,4734,1573],{"class":1091},[1050,4736,2633],{"class":1687},[1050,4738,1733],{"class":1091},[1050,4740,4741],{"class":1052,"line":1777},[1050,4742,1598],{"class":1067},[1050,4744,4745,4747,4750,4752,4754,4756,4758],{"class":1052,"line":1793},[1050,4746,1603],{"class":1565},[1050,4748,4749],{"class":1606}," is_admin_login",[1050,4751,1573],{"class":1091},[1050,4753,1613],{"class":1612},[1050,4755,1195],{"class":1091},[1050,4757,1619],{"class":1618},[1050,4759,1580],{"class":1091},[1050,4761,4762,4764,4767],{"class":1052,"line":1809},[1050,4763,1626],{"class":1585},[1050,4765,4766],{"class":1589},"Check if this is an admin login request",[1050,4768,1593],{"class":1585},[1050,4770,4771,4773],{"class":1052,"line":1814},[1050,4772,1636],{"class":1063},[1050,4774,1639],{"class":1091},[1050,4776,4777,4779,4781,4783,4785,4787,4789,4791,4793,4795,4797,4799,4801,4803,4806,4808],{"class":1052,"line":1832},[1050,4778,1644],{"class":1067},[1050,4780,1532],{"class":1091},[1050,4782,3461],{"class":1649},[1050,4784,1532],{"class":1091},[1050,4786,3466],{"class":1687},[1050,4788,1573],{"class":1091},[1050,4790,3471],{"class":1565},[1050,4792,3474],{"class":1111},[1050,4794,3478],{"class":3477},[1050,4796,1613],{"class":1087},[1050,4798,1532],{"class":1091},[1050,4800,3403],{"class":1649},[1050,4802,3487],{"class":3477},[1050,4804,4805],{"class":1111},"login/'",[1050,4807,3176],{"class":1091},[1050,4809,4810],{"class":1658}," or\n",[1050,4812,4813,4815,4817,4819,4821,4824,4826,4828,4830,4832,4834,4836],{"class":1052,"line":2433},[1050,4814,1644],{"class":1067},[1050,4816,1532],{"class":1091},[1050,4818,3461],{"class":1649},[1050,4820,2688],{"class":1658},[1050,4822,4823],{"class":1565}," f",[1050,4825,3474],{"class":1111},[1050,4827,3478],{"class":3477},[1050,4829,1613],{"class":1087},[1050,4831,1532],{"class":1091},[1050,4833,3403],{"class":1649},[1050,4835,3487],{"class":3477},[1050,4837,4838],{"class":1111},"login'\n",[1050,4840,4841],{"class":1052,"line":2454},[1050,4842,1696],{"class":1091},[1050,4844,4845],{"class":1052,"line":2470},[1050,4846,1598],{"class":1067},[1050,4848,4849,4851,4854,4856,4858,4860,4862],{"class":1052,"line":2484},[1050,4850,1603],{"class":1565},[1050,4852,4853],{"class":1606}," is_rate_limited",[1050,4855,1573],{"class":1091},[1050,4857,1613],{"class":1612},[1050,4859,1195],{"class":1091},[1050,4861,3818],{"class":1618},[1050,4863,1580],{"class":1091},[1050,4865,4866,4868,4871],{"class":1052,"line":2498},[1050,4867,1626],{"class":1585},[1050,4869,4870],{"class":1589},"Check if IP is rate limited",[1050,4872,1593],{"class":1585},[1050,4874,4875,4878,4880,4882,4885,4887,4889,4891],{"class":1052,"line":2511},[1050,4876,4877],{"class":1067},"        cache_key ",[1050,4879,1713],{"class":1091},[1050,4881,4823],{"class":1565},[1050,4883,4884],{"class":1111},"'admin_login_attempts:",[1050,4886,3478],{"class":3477},[1050,4888,3882],{"class":1067},[1050,4890,3487],{"class":3477},[1050,4892,2375],{"class":1111},[1050,4894,4895,4898,4900,4903,4905,4907,4909,4912,4914,4917],{"class":1052,"line":2522},[1050,4896,4897],{"class":1067},"        attempts ",[1050,4899,1713],{"class":1091},[1050,4901,4902],{"class":1067}," cache",[1050,4904,1532],{"class":1091},[1050,4906,2930],{"class":1687},[1050,4908,1573],{"class":1091},[1050,4910,4911],{"class":1687},"cache_key",[1050,4913,1195],{"class":1091},[1050,4915,4916],{"class":1191}," 0",[1050,4918,1733],{"class":1091},[1050,4920,4921,4923,4926,4929,4931,4933],{"class":1052,"line":2532},[1050,4922,1636],{"class":1063},[1050,4924,4925],{"class":1067}," attempts ",[1050,4927,4928],{"class":1658},">=",[1050,4930,3501],{"class":1087},[1050,4932,1532],{"class":1091},[1050,4934,4935],{"class":1649},"max_attempts\n",[1050,4937,4938],{"class":1052,"line":2538},[1050,4939,1598],{"class":1067},[1050,4941,4942,4944,4947,4949,4951,4953,4955,4957,4959],{"class":1052,"line":2544},[1050,4943,1603],{"class":1565},[1050,4945,4946],{"class":1606}," record_failed_attempt",[1050,4948,1573],{"class":1091},[1050,4950,1613],{"class":1612},[1050,4952,1195],{"class":1091},[1050,4954,3818],{"class":1618},[1050,4956,1195],{"class":1091},[1050,4958,4716],{"class":1618},[1050,4960,1580],{"class":1091},[1050,4962,4963,4965,4968],{"class":1052,"line":2549},[1050,4964,1626],{"class":1585},[1050,4966,4967],{"class":1589},"Record failed login attempt",[1050,4969,1593],{"class":1585},[1050,4971,4972,4974,4976,4978,4980,4982,4984,4986],{"class":1052,"line":2554},[1050,4973,4877],{"class":1067},[1050,4975,1713],{"class":1091},[1050,4977,4823],{"class":1565},[1050,4979,4884],{"class":1111},[1050,4981,3478],{"class":3477},[1050,4983,3882],{"class":1067},[1050,4985,3487],{"class":3477},[1050,4987,2375],{"class":1111},[1050,4989,4990,4992,4994,4996,4998,5000,5002,5004,5006,5008,5010,5012],{"class":1052,"line":2580},[1050,4991,4897],{"class":1067},[1050,4993,1713],{"class":1091},[1050,4995,4902],{"class":1067},[1050,4997,1532],{"class":1091},[1050,4999,2930],{"class":1687},[1050,5001,1573],{"class":1091},[1050,5003,4911],{"class":1687},[1050,5005,1195],{"class":1091},[1050,5007,4916],{"class":1191},[1050,5009,3176],{"class":1091},[1050,5011,2449],{"class":1658},[1050,5013,5014],{"class":1191}," 1\n",[1050,5016,5017,5020,5022,5025,5027,5029,5031,5034,5036,5038,5040,5042],{"class":1052,"line":2585},[1050,5018,5019],{"class":1067},"        cache",[1050,5021,1532],{"class":1091},[1050,5023,5024],{"class":1687},"set",[1050,5026,1573],{"class":1091},[1050,5028,4911],{"class":1687},[1050,5030,1195],{"class":1091},[1050,5032,5033],{"class":1687}," attempts",[1050,5035,1195],{"class":1091},[1050,5037,3501],{"class":1087},[1050,5039,1532],{"class":1091},[1050,5041,4410],{"class":1649},[1050,5043,1733],{"class":1091},[1050,5045,5046],{"class":1052,"line":2603},[1050,5047,2641],{"class":1067},[1050,5049,5050],{"class":1052,"line":2613},[1050,5051,5052],{"class":1056},"        # Log security event\n",[1050,5054,5055,5058,5060,5062],{"class":1052,"line":2638},[1050,5056,5057],{"class":1067},"        logger",[1050,5059,1532],{"class":1091},[1050,5061,3547],{"class":1687},[1050,5063,2065],{"class":1091},[1050,5065,5066,5069,5072,5074,5077,5079],{"class":1052,"line":2644},[1050,5067,5068],{"class":1565},"            f",[1050,5070,5071],{"class":1111},"'Failed admin login attempt #",[1050,5073,3478],{"class":3477},[1050,5075,5076],{"class":1687},"attempts",[1050,5078,3487],{"class":3477},[1050,5080,3566],{"class":1111},[1050,5082,5083,5085,5088,5090,5092,5094,5097,5099,5101,5103],{"class":1052,"line":2666},[1050,5084,5068],{"class":1565},[1050,5086,5087],{"class":1111},"'IP: ",[1050,5089,3478],{"class":3477},[1050,5091,3882],{"class":1687},[1050,5093,3487],{"class":3477},[1050,5095,5096],{"class":1111},", Username: ",[1050,5098,3478],{"class":3477},[1050,5100,2323],{"class":1687},[1050,5102,3487],{"class":3477},[1050,5104,2375],{"class":1111},[1050,5106,5107],{"class":1052,"line":2672},[1050,5108,1696],{"class":1091},[1050,5110,5111],{"class":1052,"line":2699},[1050,5112,2641],{"class":1067},[1050,5114,5115],{"class":1052,"line":2717},[1050,5116,5117],{"class":1056},"        # Lock user account after multiple failures\n",[1050,5119,5120,5122,5124,5126,5129],{"class":1052,"line":2742},[1050,5121,2647],{"class":1063},[1050,5123,4925],{"class":1067},[1050,5125,4928],{"class":1658},[1050,5127,5128],{"class":1191}," 3",[1050,5130,2663],{"class":1091},[1050,5132,5133,5136],{"class":1052,"line":2748},[1050,5134,5135],{"class":1063},"            try",[1050,5137,2663],{"class":1091},[1050,5139,5140,5143,5145,5147,5149,5151,5153,5155,5157,5159],{"class":1052,"line":2774},[1050,5141,5142],{"class":1063},"                from",[1050,5144,1749],{"class":1067},[1050,5146,1532],{"class":1091},[1050,5148,1754],{"class":1067},[1050,5150,1532],{"class":1091},[1050,5152,1759],{"class":1067},[1050,5154,1532],{"class":1091},[1050,5156,1764],{"class":1067},[1050,5158,1064],{"class":1063},[1050,5160,5161],{"class":1067}," User\n",[1050,5163,5164,5167,5169,5171,5173,5176,5178,5180,5182,5184,5186,5188],{"class":1052,"line":2801},[1050,5165,5166],{"class":1067},"                user ",[1050,5168,1713],{"class":1091},[1050,5170,1769],{"class":1067},[1050,5172,1532],{"class":1091},[1050,5174,5175],{"class":1649},"objects",[1050,5177,1532],{"class":1091},[1050,5179,2930],{"class":1687},[1050,5181,1573],{"class":1091},[1050,5183,2323],{"class":1720},[1050,5185,1713],{"class":1091},[1050,5187,2323],{"class":1687},[1050,5189,1733],{"class":1091},[1050,5191,5192,5195,5197,5199,5201],{"class":1052,"line":2806},[1050,5193,5194],{"class":1067},"                user",[1050,5196,1532],{"class":1091},[1050,5198,2372],{"class":1649},[1050,5200,1092],{"class":1091},[1050,5202,5203],{"class":1067}," attempts\n",[1050,5205,5206,5208,5210,5212,5214,5216,5218],{"class":1052,"line":2814},[1050,5207,4050],{"class":1063},[1050,5209,4925],{"class":1067},[1050,5211,4928],{"class":1658},[1050,5213,3501],{"class":1087},[1050,5215,1532],{"class":1091},[1050,5217,4396],{"class":1649},[1050,5219,2663],{"class":1091},[1050,5221,5222,5225,5227,5229,5231,5234,5236,5239,5242,5244,5247],{"class":1052,"line":2819},[1050,5223,5224],{"class":1067},"                    user",[1050,5226,1532],{"class":1091},[1050,5228,2527],{"class":1649},[1050,5230,1092],{"class":1091},[1050,5232,5233],{"class":1067}," timezone",[1050,5235,1532],{"class":1091},[1050,5237,5238],{"class":1687},"now",[1050,5240,5241],{"class":1091},"()",[1050,5243,2449],{"class":1658},[1050,5245,5246],{"class":1687}," timedelta",[1050,5248,2065],{"class":1091},[1050,5250,5251,5254,5256,5258,5260],{"class":1052,"line":4159},[1050,5252,5253],{"class":1720},"                        seconds",[1050,5255,1713],{"class":1091},[1050,5257,1613],{"class":1087},[1050,5259,1532],{"class":1091},[1050,5261,5262],{"class":1649},"lockout_duration\n",[1050,5264,5265],{"class":1052,"line":4168},[1050,5266,5267],{"class":1091},"                    )\n",[1050,5269,5270,5272,5274,5277],{"class":1052,"line":4180},[1050,5271,5194],{"class":1067},[1050,5273,1532],{"class":1091},[1050,5275,5276],{"class":1687},"save",[1050,5278,1691],{"class":1091},[1050,5280,5281,5284,5286,5288,5291],{"class":1052,"line":4185},[1050,5282,5283],{"class":1063},"            except",[1050,5285,1769],{"class":1067},[1050,5287,1532],{"class":1091},[1050,5289,5290],{"class":1649},"DoesNotExist",[1050,5292,2663],{"class":1091},[1050,5294,5295],{"class":1052,"line":4190},[1050,5296,5297],{"class":1063},"                pass\n",[1030,5299,5301],{"id":5300},"data-protection-and-validation","Data Protection and Validation",[1035,5303,5305],{"id":5304},"input-sanitization","Input Sanitization",[1040,5307,5309],{"className":1042,"code":5308,"language":1044,"meta":1045,"style":1045},"# admin.py\nfrom django.contrib import admin\nfrom django.core.exceptions import ValidationError\nfrom django.utils.html import strip_tags\nimport bleach\n\nclass SecureModelAdmin(admin.ModelAdmin):\n    \"\"\"Base admin class with input sanitization\"\"\"\n    \n    def clean_html_fields(self, obj):\n        \"\"\"Sanitize HTML content in specified fields\"\"\"\n        html_fields = getattr(self, 'html_fields', [])\n        \n        for field_name in html_fields:\n            if hasattr(obj, field_name):\n                content = getattr(obj, field_name)\n                if content:\n                    # Allow only safe HTML tags\n                    allowed_tags = [\n                        'p', 'br', 'strong', 'em', 'u', 'ol', 'ul', 'li',\n                        'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'blockquote'\n                    ]\n                    \n                    cleaned_content = bleach.clean(\n                        content,\n                        tags=allowed_tags,\n                        strip=True\n                    )\n                    \n                    setattr(obj, field_name, cleaned_content)\n    \n    def save_model(self, request, obj, form, change):\n        \"\"\"Override save to add security checks\"\"\"\n        # Sanitize HTML fields\n        self.clean_html_fields(obj)\n        \n        # Log admin actions\n        action = 'changed' if change else 'added'\n        logger.info(\n            f'Admin action: {request.user.username} {action} '\n            f'{obj._meta.model_name} \"{obj}\"'\n        )\n        \n        super().save_model(request, obj, form, change)\n\n@admin.register(Article)\nclass ArticleAdmin(SecureModelAdmin):\n    html_fields = ['content', 'excerpt']  # Fields to sanitize\n",[1047,5310,5311,5315,5329,5349,5370,5377,5381,5399,5408,5412,5430,5439,5465,5469,5483,5502,5521,5530,5535,5544,5618,5678,5683,5688,5705,5712,5724,5733,5737,5741,5761,5765,5796,5805,5810,5825,5829,5834,5864,5875,5907,5940,5944,5948,5976,5980,5998,6012],{"__ignoreMap":1045},[1050,5312,5313],{"class":1052,"line":1053},[1050,5314,2221],{"class":1056},[1050,5316,5317,5319,5321,5323,5325,5327],{"class":1052,"line":1060},[1050,5318,1526],{"class":1063},[1050,5320,1749],{"class":1067},[1050,5322,1532],{"class":1091},[1050,5324,2232],{"class":1067},[1050,5326,1064],{"class":1063},[1050,5328,2237],{"class":1067},[1050,5330,5331,5333,5335,5337,5339,5341,5344,5346],{"class":1052,"line":1071},[1050,5332,1526],{"class":1063},[1050,5334,1749],{"class":1067},[1050,5336,1532],{"class":1091},[1050,5338,4281],{"class":1067},[1050,5340,1532],{"class":1091},[1050,5342,5343],{"class":1067},"exceptions ",[1050,5345,1064],{"class":1063},[1050,5347,5348],{"class":1067}," ValidationError\n",[1050,5350,5351,5353,5355,5357,5360,5362,5365,5367],{"class":1052,"line":1078},[1050,5352,1526],{"class":1063},[1050,5354,1749],{"class":1067},[1050,5356,1532],{"class":1091},[1050,5358,5359],{"class":1067},"utils",[1050,5361,1532],{"class":1091},[1050,5363,5364],{"class":1067},"html ",[1050,5366,1064],{"class":1063},[1050,5368,5369],{"class":1067}," strip_tags\n",[1050,5371,5372,5374],{"class":1052,"line":1084},[1050,5373,1064],{"class":1063},[1050,5375,5376],{"class":1067}," bleach\n",[1050,5378,5379],{"class":1052,"line":1098},[1050,5380,1075],{"emptyLinePlaceholder":1074},[1050,5382,5383,5385,5388,5390,5392,5394,5397],{"class":1052,"line":1104},[1050,5384,1566],{"class":1565},[1050,5386,5387],{"class":1569}," SecureModelAdmin",[1050,5389,1573],{"class":1091},[1050,5391,2822],{"class":1576},[1050,5393,1532],{"class":1091},[1050,5395,5396],{"class":1576},"ModelAdmin",[1050,5398,1580],{"class":1091},[1050,5400,5401,5403,5406],{"class":1052,"line":1132},[1050,5402,1586],{"class":1585},[1050,5404,5405],{"class":1589},"Base admin class with input sanitization",[1050,5407,1593],{"class":1585},[1050,5409,5410],{"class":1052,"line":1138},[1050,5411,1598],{"class":1067},[1050,5413,5414,5416,5419,5421,5423,5425,5428],{"class":1052,"line":1143},[1050,5415,1603],{"class":1565},[1050,5417,5418],{"class":1606}," clean_html_fields",[1050,5420,1573],{"class":1091},[1050,5422,1613],{"class":1612},[1050,5424,1195],{"class":1091},[1050,5426,5427],{"class":1618}," obj",[1050,5429,1580],{"class":1091},[1050,5431,5432,5434,5437],{"class":1052,"line":1163},[1050,5433,1626],{"class":1585},[1050,5435,5436],{"class":1589},"Sanitize HTML content in specified fields",[1050,5438,1593],{"class":1585},[1050,5440,5441,5444,5446,5448,5450,5452,5454,5456,5459,5461,5463],{"class":1052,"line":1178},[1050,5442,5443],{"class":1067},"        html_fields ",[1050,5445,1713],{"class":1091},[1050,5447,3375],{"class":3188},[1050,5449,1573],{"class":1091},[1050,5451,1613],{"class":1087},[1050,5453,1195],{"class":1091},[1050,5455,1121],{"class":1107},[1050,5457,5458],{"class":1111},"html_fields",[1050,5460,1115],{"class":1107},[1050,5462,1195],{"class":1091},[1050,5464,3394],{"class":1091},[1050,5466,5467],{"class":1052,"line":1201},[1050,5468,2641],{"class":1067},[1050,5470,5471,5473,5476,5478,5481],{"class":1052,"line":1207},[1050,5472,3775],{"class":1063},[1050,5474,5475],{"class":1067}," field_name ",[1050,5477,3185],{"class":1063},[1050,5479,5480],{"class":1067}," html_fields",[1050,5482,2663],{"class":1091},[1050,5484,5485,5487,5490,5492,5495,5497,5500],{"class":1052,"line":1212},[1050,5486,2675],{"class":1063},[1050,5488,5489],{"class":3188}," hasattr",[1050,5491,1573],{"class":1091},[1050,5493,5494],{"class":1687},"obj",[1050,5496,1195],{"class":1091},[1050,5498,5499],{"class":1687}," field_name",[1050,5501,1580],{"class":1091},[1050,5503,5504,5507,5509,5511,5513,5515,5517,5519],{"class":1052,"line":1217},[1050,5505,5506],{"class":1067},"                content ",[1050,5508,1713],{"class":1091},[1050,5510,3375],{"class":3188},[1050,5512,1573],{"class":1091},[1050,5514,5494],{"class":1687},[1050,5516,1195],{"class":1091},[1050,5518,5499],{"class":1687},[1050,5520,1733],{"class":1091},[1050,5522,5523,5525,5528],{"class":1052,"line":1237},[1050,5524,4050],{"class":1063},[1050,5526,5527],{"class":1067}," content",[1050,5529,2663],{"class":1091},[1050,5531,5532],{"class":1052,"line":1242},[1050,5533,5534],{"class":1056},"                    # Allow only safe HTML tags\n",[1050,5536,5537,5540,5542],{"class":1052,"line":1247},[1050,5538,5539],{"class":1067},"                    allowed_tags ",[1050,5541,1713],{"class":1091},[1050,5543,1095],{"class":1091},[1050,5545,5546,5549,5551,5553,5555,5557,5560,5562,5564,5566,5569,5571,5573,5575,5578,5580,5582,5584,5587,5589,5591,5593,5596,5598,5600,5602,5605,5607,5609,5611,5614,5616],{"class":1052,"line":1267},[1050,5547,5548],{"class":1107},"                        '",[1050,5550,1026],{"class":1111},[1050,5552,1115],{"class":1107},[1050,5554,1195],{"class":1091},[1050,5556,1121],{"class":1107},[1050,5558,5559],{"class":1111},"br",[1050,5561,1115],{"class":1107},[1050,5563,1195],{"class":1091},[1050,5565,1121],{"class":1107},[1050,5567,5568],{"class":1111},"strong",[1050,5570,1115],{"class":1107},[1050,5572,1195],{"class":1091},[1050,5574,1121],{"class":1107},[1050,5576,5577],{"class":1111},"em",[1050,5579,1115],{"class":1107},[1050,5581,1195],{"class":1091},[1050,5583,1121],{"class":1107},[1050,5585,5586],{"class":1111},"u",[1050,5588,1115],{"class":1107},[1050,5590,1195],{"class":1091},[1050,5592,1121],{"class":1107},[1050,5594,5595],{"class":1111},"ol",[1050,5597,1115],{"class":1107},[1050,5599,1195],{"class":1091},[1050,5601,1121],{"class":1107},[1050,5603,5604],{"class":1111},"ul",[1050,5606,1115],{"class":1107},[1050,5608,1195],{"class":1091},[1050,5610,1121],{"class":1107},[1050,5612,5613],{"class":1111},"li",[1050,5615,1115],{"class":1107},[1050,5617,1129],{"class":1091},[1050,5619,5620,5622,5624,5626,5628,5630,5632,5634,5636,5638,5640,5642,5644,5646,5649,5651,5653,5655,5658,5660,5662,5664,5667,5669,5671,5673,5676],{"class":1052,"line":1272},[1050,5621,5548],{"class":1107},[1050,5623,1022],{"class":1111},[1050,5625,1115],{"class":1107},[1050,5627,1195],{"class":1091},[1050,5629,1121],{"class":1107},[1050,5631,1030],{"class":1111},[1050,5633,1115],{"class":1107},[1050,5635,1195],{"class":1091},[1050,5637,1121],{"class":1107},[1050,5639,1035],{"class":1111},[1050,5641,1115],{"class":1107},[1050,5643,1195],{"class":1091},[1050,5645,1121],{"class":1107},[1050,5647,5648],{"class":1111},"h4",[1050,5650,1115],{"class":1107},[1050,5652,1195],{"class":1091},[1050,5654,1121],{"class":1107},[1050,5656,5657],{"class":1111},"h5",[1050,5659,1115],{"class":1107},[1050,5661,1195],{"class":1091},[1050,5663,1121],{"class":1107},[1050,5665,5666],{"class":1111},"h6",[1050,5668,1115],{"class":1107},[1050,5670,1195],{"class":1091},[1050,5672,1121],{"class":1107},[1050,5674,5675],{"class":1111},"blockquote",[1050,5677,2375],{"class":1107},[1050,5679,5680],{"class":1052,"line":1278},[1050,5681,5682],{"class":1091},"                    ]\n",[1050,5684,5685],{"class":1052,"line":1283},[1050,5686,5687],{"class":1067},"                    \n",[1050,5689,5690,5693,5695,5698,5700,5703],{"class":1052,"line":1289},[1050,5691,5692],{"class":1067},"                    cleaned_content ",[1050,5694,1713],{"class":1091},[1050,5696,5697],{"class":1067}," bleach",[1050,5699,1532],{"class":1091},[1050,5701,5702],{"class":1687},"clean",[1050,5704,2065],{"class":1091},[1050,5706,5707,5710],{"class":1052,"line":1304},[1050,5708,5709],{"class":1687},"                        content",[1050,5711,1129],{"class":1091},[1050,5713,5714,5717,5719,5722],{"class":1052,"line":1317},[1050,5715,5716],{"class":1720},"                        tags",[1050,5718,1713],{"class":1091},[1050,5720,5721],{"class":1687},"allowed_tags",[1050,5723,1129],{"class":1091},[1050,5725,5726,5729,5731],{"class":1052,"line":1331},[1050,5727,5728],{"class":1720},"                        strip",[1050,5730,1713],{"class":1091},[1050,5732,2112],{"class":1297},[1050,5734,5735],{"class":1052,"line":1342},[1050,5736,5267],{"class":1091},[1050,5738,5739],{"class":1052,"line":1347},[1050,5740,5687],{"class":1067},[1050,5742,5743,5746,5748,5750,5752,5754,5756,5759],{"class":1052,"line":1353},[1050,5744,5745],{"class":3188},"                    setattr",[1050,5747,1573],{"class":1091},[1050,5749,5494],{"class":1687},[1050,5751,1195],{"class":1091},[1050,5753,5499],{"class":1687},[1050,5755,1195],{"class":1091},[1050,5757,5758],{"class":1687}," cleaned_content",[1050,5760,1733],{"class":1091},[1050,5762,5763],{"class":1052,"line":1363},[1050,5764,1598],{"class":1067},[1050,5766,5767,5769,5772,5774,5776,5778,5780,5782,5784,5786,5789,5791,5794],{"class":1052,"line":1373},[1050,5768,1603],{"class":1565},[1050,5770,5771],{"class":1606}," save_model",[1050,5773,1573],{"class":1091},[1050,5775,1613],{"class":1612},[1050,5777,1195],{"class":1091},[1050,5779,1619],{"class":1618},[1050,5781,1195],{"class":1091},[1050,5783,5427],{"class":1618},[1050,5785,1195],{"class":1091},[1050,5787,5788],{"class":1618}," form",[1050,5790,1195],{"class":1091},[1050,5792,5793],{"class":1618}," change",[1050,5795,1580],{"class":1091},[1050,5797,5798,5800,5803],{"class":1052,"line":1383},[1050,5799,1626],{"class":1585},[1050,5801,5802],{"class":1589},"Override save to add security checks",[1050,5804,1593],{"class":1585},[1050,5806,5807],{"class":1052,"line":1388},[1050,5808,5809],{"class":1056},"        # Sanitize HTML fields\n",[1050,5811,5812,5814,5816,5819,5821,5823],{"class":1052,"line":1394},[1050,5813,3351],{"class":1087},[1050,5815,1532],{"class":1091},[1050,5817,5818],{"class":1687},"clean_html_fields",[1050,5820,1573],{"class":1091},[1050,5822,5494],{"class":1687},[1050,5824,1733],{"class":1091},[1050,5826,5827],{"class":1052,"line":1777},[1050,5828,2641],{"class":1067},[1050,5830,5831],{"class":1052,"line":1793},[1050,5832,5833],{"class":1056},"        # Log admin actions\n",[1050,5835,5836,5839,5841,5843,5846,5848,5851,5854,5857,5859,5862],{"class":1052,"line":1809},[1050,5837,5838],{"class":1067},"        action ",[1050,5840,1713],{"class":1091},[1050,5842,1121],{"class":1107},[1050,5844,5845],{"class":1111},"changed",[1050,5847,1115],{"class":1107},[1050,5849,5850],{"class":1063}," if",[1050,5852,5853],{"class":1067}," change ",[1050,5855,5856],{"class":1063},"else",[1050,5858,1121],{"class":1107},[1050,5860,5861],{"class":1111},"added",[1050,5863,2375],{"class":1107},[1050,5865,5866,5868,5870,5873],{"class":1052,"line":1814},[1050,5867,5057],{"class":1067},[1050,5869,1532],{"class":1091},[1050,5871,5872],{"class":1687},"info",[1050,5874,2065],{"class":1091},[1050,5876,5877,5879,5882,5884,5886,5888,5890,5892,5894,5896,5899,5902,5904],{"class":1052,"line":1832},[1050,5878,5068],{"class":1565},[1050,5880,5881],{"class":1111},"'Admin action: ",[1050,5883,3478],{"class":3477},[1050,5885,2633],{"class":1687},[1050,5887,1532],{"class":1091},[1050,5889,1650],{"class":1649},[1050,5891,1532],{"class":1091},[1050,5893,2323],{"class":1649},[1050,5895,3487],{"class":3477},[1050,5897,5898],{"class":3477}," {",[1050,5900,5901],{"class":1687},"action",[1050,5903,3487],{"class":3477},[1050,5905,5906],{"class":1111}," '\n",[1050,5908,5909,5911,5913,5915,5917,5919,5922,5924,5927,5929,5931,5933,5935,5937],{"class":1052,"line":2433},[1050,5910,5068],{"class":1565},[1050,5912,1115],{"class":1111},[1050,5914,3478],{"class":3477},[1050,5916,5494],{"class":1687},[1050,5918,1532],{"class":1091},[1050,5920,5921],{"class":1649},"_meta",[1050,5923,1532],{"class":1091},[1050,5925,5926],{"class":1649},"model_name",[1050,5928,3487],{"class":3477},[1050,5930,3591],{"class":1111},[1050,5932,3478],{"class":3477},[1050,5934,5494],{"class":1687},[1050,5936,3487],{"class":3477},[1050,5938,5939],{"class":1111},"\"'\n",[1050,5941,5942],{"class":1052,"line":2454},[1050,5943,1696],{"class":1091},[1050,5945,5946],{"class":1052,"line":2470},[1050,5947,2641],{"class":1067},[1050,5949,5950,5953,5955,5958,5960,5962,5964,5966,5968,5970,5972,5974],{"class":1052,"line":2484},[1050,5951,5952],{"class":2621},"        super",[1050,5954,2625],{"class":1091},[1050,5956,5957],{"class":1687},"save_model",[1050,5959,1573],{"class":1091},[1050,5961,2633],{"class":1687},[1050,5963,1195],{"class":1091},[1050,5965,5427],{"class":1687},[1050,5967,1195],{"class":1091},[1050,5969,5788],{"class":1687},[1050,5971,1195],{"class":1091},[1050,5973,5793],{"class":1687},[1050,5975,1733],{"class":1091},[1050,5977,5978],{"class":1052,"line":2498},[1050,5979,1075],{"emptyLinePlaceholder":1074},[1050,5981,5982,5985,5987,5989,5991,5993,5996],{"class":1052,"line":2511},[1050,5983,5984],{"class":1091},"@",[1050,5986,2822],{"class":1606},[1050,5988,1532],{"class":1091},[1050,5990,1822],{"class":1606},[1050,5992,1573],{"class":1091},[1050,5994,5995],{"class":1687},"Article",[1050,5997,1733],{"class":1091},[1050,5999,6000,6002,6005,6007,6010],{"class":1052,"line":2522},[1050,6001,1566],{"class":1565},[1050,6003,6004],{"class":1569}," ArticleAdmin",[1050,6006,1573],{"class":1091},[1050,6008,6009],{"class":1576},"SecureModelAdmin",[1050,6011,1580],{"class":1091},[1050,6013,6014,6017,6019,6021,6023,6026,6028,6030,6032,6035,6037,6040],{"class":1052,"line":2532},[1050,6015,6016],{"class":1067},"    html_fields ",[1050,6018,1713],{"class":1091},[1050,6020,2393],{"class":1091},[1050,6022,1115],{"class":1107},[1050,6024,6025],{"class":1111},"content",[1050,6027,1115],{"class":1107},[1050,6029,1195],{"class":1091},[1050,6031,1121],{"class":1107},[1050,6033,6034],{"class":1111},"excerpt",[1050,6036,1115],{"class":1107},[1050,6038,6039],{"class":1091},"]",[1050,6041,6042],{"class":1056},"  # Fields to sanitize\n",[1035,6044,6046],{"id":6045},"file-upload-security","File Upload Security",[1040,6048,6050],{"className":1042,"code":6049,"language":1044,"meta":1045,"style":1045},"# validators.py\nimport magic\nimport os\nfrom django.core.exceptions import ValidationError\nfrom django.conf import settings\n\nclass SecureFileValidator:\n    \"\"\"Comprehensive file upload validation\"\"\"\n    \n    def __init__(self, allowed_types=None, max_size=None, scan_content=True):\n        self.allowed_types = allowed_types or []\n        self.max_size = max_size or 10 * 1024 * 1024  # 10MB default\n        self.scan_content = scan_content\n    \n    def __call__(self, file):\n        self.validate_size(file)\n        self.validate_type(file)\n        self.validate_filename(file)\n        \n        if self.scan_content:\n            self.scan_for_malware(file)\n    \n    def validate_size(self, file):\n        \"\"\"Validate file size\"\"\"\n        if file.size > self.max_size:\n            raise ValidationError(\n                f'File size {file.size} exceeds maximum allowed size {self.max_size}'\n            )\n    \n    def validate_type(self, file):\n        \"\"\"Validate file type using magic numbers\"\"\"\n        file.seek(0)\n        file_header = file.read(1024)\n        file.seek(0)\n        \n        detected_type = magic.from_buffer(file_header, mime=True)\n        \n        if self.allowed_types and detected_type not in self.allowed_types:\n            raise ValidationError(\n                f'File type {detected_type} is not allowed'\n            )\n    \n    def validate_filename(self, file):\n        \"\"\"Validate and sanitize filename\"\"\"\n        filename = file.name\n        \n        # Check for dangerous extensions\n        dangerous_extensions = [\n            '.exe', '.bat', '.cmd', '.scr', '.pif', '.com',\n            '.php', '.asp', '.jsp', '.js', '.vbs', '.sh'\n        ]\n        \n        file_ext = os.path.splitext(filename)[1].lower()\n        if file_ext in dangerous_extensions:\n            raise ValidationError(f'File extension {file_ext} is not allowed')\n        \n        # Check for path traversal attempts\n        if '..' in filename or '/' in filename or '\\\\' in filename:\n            raise ValidationError('Invalid filename')\n    \n    def scan_for_malware(self, file):\n        \"\"\"Basic malware signature detection\"\"\"\n        file.seek(0)\n        content = file.read(8192)  # Read first 8KB\n        file.seek(0)\n        \n        # Check for suspicious patterns\n        malicious_patterns = [\n            b'\u003Cscript',\n            b'javascript:',\n            b'\u003C?php',\n            b'\u003C%',\n            b'eval(',\n            b'exec(',\n            b'system(',\n            b'shell_exec(',\n        ]\n        \n        content_lower = content.lower()\n        for pattern in malicious_patterns:\n            if pattern in content_lower:\n                raise ValidationError('Potentially malicious content detected')\n\n# Usage in admin\nsecure_image_validator = SecureFileValidator(\n    allowed_types=['image/jpeg', 'image/png', 'image/gif'],\n    max_size=5 * 1024 * 1024  # 5MB\n)\n\nclass MediaFileAdmin(admin.ModelAdmin):\n    def formfield_for_dbfield(self, db_field, **kwargs):\n        \"\"\"Add validators to file fields\"\"\"\n        if db_field.name in ['image', 'file']:\n            kwargs['validators'] = [secure_image_validator]\n        \n        return super().formfield_for_dbfield(db_field, **kwargs)\n",[1047,6051,6052,6057,6064,6070,6088,6102,6106,6115,6124,6128,6168,6188,6220,6234,6238,6255,6272,6287,6302,6306,6318,6334,6338,6355,6364,6386,6396,6429,6433,6437,6454,6463,6479,6500,6514,6518,6549,6553,6582,6590,6607,6611,6615,6632,6641,6655,6659,6664,6673,6729,6783,6787,6791,6826,6840,6865,6869,6874,6918,6935,6939,6956,6965,6979,7002,7016,7020,7025,7034,7048,7061,7074,7087,7100,7113,7126,7139,7143,7147,7162,7176,7190,7209,7214,7220,7232,7268,7290,7295,7300,7318,7345,7355,7387,7414,7419],{"__ignoreMap":1045},[1050,6053,6054],{"class":1052,"line":1053},[1050,6055,6056],{"class":1056},"# validators.py\n",[1050,6058,6059,6061],{"class":1052,"line":1060},[1050,6060,1064],{"class":1063},[1050,6062,6063],{"class":1067}," magic\n",[1050,6065,6066,6068],{"class":1052,"line":1071},[1050,6067,1064],{"class":1063},[1050,6069,1068],{"class":1067},[1050,6071,6072,6074,6076,6078,6080,6082,6084,6086],{"class":1052,"line":1078},[1050,6073,1526],{"class":1063},[1050,6075,1749],{"class":1067},[1050,6077,1532],{"class":1091},[1050,6079,4281],{"class":1067},[1050,6081,1532],{"class":1091},[1050,6083,5343],{"class":1067},[1050,6085,1064],{"class":1063},[1050,6087,5348],{"class":1067},[1050,6089,6090,6092,6094,6096,6098,6100],{"class":1052,"line":1084},[1050,6091,1526],{"class":1063},[1050,6093,1749],{"class":1067},[1050,6095,1532],{"class":1091},[1050,6097,3256],{"class":1067},[1050,6099,1064],{"class":1063},[1050,6101,3261],{"class":1067},[1050,6103,6104],{"class":1052,"line":1098},[1050,6105,1075],{"emptyLinePlaceholder":1074},[1050,6107,6108,6110,6113],{"class":1052,"line":1104},[1050,6109,1566],{"class":1565},[1050,6111,6112],{"class":1569}," SecureFileValidator",[1050,6114,2663],{"class":1091},[1050,6116,6117,6119,6122],{"class":1052,"line":1132},[1050,6118,1586],{"class":1585},[1050,6120,6121],{"class":1589},"Comprehensive file upload validation",[1050,6123,1593],{"class":1585},[1050,6125,6126],{"class":1052,"line":1138},[1050,6127,1598],{"class":1067},[1050,6129,6130,6132,6134,6136,6138,6140,6143,6145,6148,6150,6153,6155,6157,6159,6162,6164,6166],{"class":1052,"line":1143},[1050,6131,1603],{"class":1565},[1050,6133,3335],{"class":3188},[1050,6135,1573],{"class":1091},[1050,6137,1613],{"class":1612},[1050,6139,1195],{"class":1091},[1050,6141,6142],{"class":1618}," allowed_types",[1050,6144,1713],{"class":1658},[1050,6146,6147],{"class":1297},"None",[1050,6149,1195],{"class":1091},[1050,6151,6152],{"class":1618}," max_size",[1050,6154,1713],{"class":1658},[1050,6156,6147],{"class":1297},[1050,6158,1195],{"class":1091},[1050,6160,6161],{"class":1618}," scan_content",[1050,6163,1713],{"class":1658},[1050,6165,2100],{"class":1297},[1050,6167,1580],{"class":1091},[1050,6169,6170,6172,6174,6177,6179,6182,6185],{"class":1052,"line":1163},[1050,6171,3351],{"class":1087},[1050,6173,1532],{"class":1091},[1050,6175,6176],{"class":1649},"allowed_types",[1050,6178,1092],{"class":1091},[1050,6180,6181],{"class":1067}," allowed_types ",[1050,6183,6184],{"class":1658},"or",[1050,6186,6187],{"class":1091}," []\n",[1050,6189,6190,6192,6194,6197,6199,6202,6204,6207,6210,6213,6215,6217],{"class":1052,"line":1178},[1050,6191,3351],{"class":1087},[1050,6193,1532],{"class":1091},[1050,6195,6196],{"class":1649},"max_size",[1050,6198,1092],{"class":1091},[1050,6200,6201],{"class":1067}," max_size ",[1050,6203,6184],{"class":1658},[1050,6205,6206],{"class":1191}," 10",[1050,6208,6209],{"class":1658}," *",[1050,6211,6212],{"class":1191}," 1024",[1050,6214,6209],{"class":1658},[1050,6216,6212],{"class":1191},[1050,6218,6219],{"class":1056},"  # 10MB default\n",[1050,6221,6222,6224,6226,6229,6231],{"class":1052,"line":1201},[1050,6223,3351],{"class":1087},[1050,6225,1532],{"class":1091},[1050,6227,6228],{"class":1649},"scan_content",[1050,6230,1092],{"class":1091},[1050,6232,6233],{"class":1067}," scan_content\n",[1050,6235,6236],{"class":1052,"line":1207},[1050,6237,1598],{"class":1067},[1050,6239,6240,6242,6244,6246,6248,6250,6253],{"class":1052,"line":1212},[1050,6241,1603],{"class":1565},[1050,6243,3440],{"class":3188},[1050,6245,1573],{"class":1091},[1050,6247,1613],{"class":1612},[1050,6249,1195],{"class":1091},[1050,6251,6252],{"class":1618}," file",[1050,6254,1580],{"class":1091},[1050,6256,6257,6259,6261,6264,6266,6270],{"class":1052,"line":1217},[1050,6258,3351],{"class":1087},[1050,6260,1532],{"class":1091},[1050,6262,6263],{"class":1687},"validate_size",[1050,6265,1573],{"class":1091},[1050,6267,6269],{"class":6268},"sSC40","file",[1050,6271,1733],{"class":1091},[1050,6273,6274,6276,6278,6281,6283,6285],{"class":1052,"line":1237},[1050,6275,3351],{"class":1087},[1050,6277,1532],{"class":1091},[1050,6279,6280],{"class":1687},"validate_type",[1050,6282,1573],{"class":1091},[1050,6284,6269],{"class":6268},[1050,6286,1733],{"class":1091},[1050,6288,6289,6291,6293,6296,6298,6300],{"class":1052,"line":1242},[1050,6290,3351],{"class":1087},[1050,6292,1532],{"class":1091},[1050,6294,6295],{"class":1687},"validate_filename",[1050,6297,1573],{"class":1091},[1050,6299,6269],{"class":6268},[1050,6301,1733],{"class":1091},[1050,6303,6304],{"class":1052,"line":1247},[1050,6305,2641],{"class":1067},[1050,6307,6308,6310,6312,6314,6316],{"class":1052,"line":1267},[1050,6309,2647],{"class":1063},[1050,6311,3501],{"class":1087},[1050,6313,1532],{"class":1091},[1050,6315,6228],{"class":1649},[1050,6317,2663],{"class":1091},[1050,6319,6320,6323,6325,6328,6330,6332],{"class":1052,"line":1272},[1050,6321,6322],{"class":1087},"            self",[1050,6324,1532],{"class":1091},[1050,6326,6327],{"class":1687},"scan_for_malware",[1050,6329,1573],{"class":1091},[1050,6331,6269],{"class":6268},[1050,6333,1733],{"class":1091},[1050,6335,6336],{"class":1052,"line":1278},[1050,6337,1598],{"class":1067},[1050,6339,6340,6342,6345,6347,6349,6351,6353],{"class":1052,"line":1283},[1050,6341,1603],{"class":1565},[1050,6343,6344],{"class":1606}," validate_size",[1050,6346,1573],{"class":1091},[1050,6348,1613],{"class":1612},[1050,6350,1195],{"class":1091},[1050,6352,6252],{"class":1618},[1050,6354,1580],{"class":1091},[1050,6356,6357,6359,6362],{"class":1052,"line":1289},[1050,6358,1626],{"class":1585},[1050,6360,6361],{"class":1589},"Validate file size",[1050,6363,1593],{"class":1585},[1050,6365,6366,6368,6370,6372,6375,6378,6380,6382,6384],{"class":1052,"line":1304},[1050,6367,2647],{"class":1063},[1050,6369,6252],{"class":6268},[1050,6371,1532],{"class":1091},[1050,6373,6374],{"class":1649},"size",[1050,6376,6377],{"class":1658}," >",[1050,6379,3501],{"class":1087},[1050,6381,1532],{"class":1091},[1050,6383,6196],{"class":1649},[1050,6385,2663],{"class":1091},[1050,6387,6388,6391,6394],{"class":1052,"line":1317},[1050,6389,6390],{"class":1063},"            raise",[1050,6392,6393],{"class":1687}," ValidationError",[1050,6395,2065],{"class":1091},[1050,6397,6398,6401,6404,6406,6408,6410,6412,6414,6417,6419,6421,6423,6425,6427],{"class":1052,"line":1331},[1050,6399,6400],{"class":1565},"                f",[1050,6402,6403],{"class":1111},"'File size ",[1050,6405,3478],{"class":3477},[1050,6407,6269],{"class":6268},[1050,6409,1532],{"class":1091},[1050,6411,6374],{"class":1649},[1050,6413,3487],{"class":3477},[1050,6415,6416],{"class":1111}," exceeds maximum allowed size ",[1050,6418,3478],{"class":3477},[1050,6420,1613],{"class":1087},[1050,6422,1532],{"class":1091},[1050,6424,6196],{"class":1649},[1050,6426,3487],{"class":3477},[1050,6428,2375],{"class":1111},[1050,6430,6431],{"class":1052,"line":1342},[1050,6432,2535],{"class":1091},[1050,6434,6435],{"class":1052,"line":1347},[1050,6436,1598],{"class":1067},[1050,6438,6439,6441,6444,6446,6448,6450,6452],{"class":1052,"line":1353},[1050,6440,1603],{"class":1565},[1050,6442,6443],{"class":1606}," validate_type",[1050,6445,1573],{"class":1091},[1050,6447,1613],{"class":1612},[1050,6449,1195],{"class":1091},[1050,6451,6252],{"class":1618},[1050,6453,1580],{"class":1091},[1050,6455,6456,6458,6461],{"class":1052,"line":1363},[1050,6457,1626],{"class":1585},[1050,6459,6460],{"class":1589},"Validate file type using magic numbers",[1050,6462,1593],{"class":1585},[1050,6464,6465,6468,6470,6473,6475,6477],{"class":1052,"line":1373},[1050,6466,6467],{"class":6268},"        file",[1050,6469,1532],{"class":1091},[1050,6471,6472],{"class":1687},"seek",[1050,6474,1573],{"class":1091},[1050,6476,2179],{"class":1191},[1050,6478,1733],{"class":1091},[1050,6480,6481,6484,6486,6488,6490,6493,6495,6498],{"class":1052,"line":1383},[1050,6482,6483],{"class":1067},"        file_header ",[1050,6485,1713],{"class":1091},[1050,6487,6252],{"class":6268},[1050,6489,1532],{"class":1091},[1050,6491,6492],{"class":1687},"read",[1050,6494,1573],{"class":1091},[1050,6496,6497],{"class":1191},"1024",[1050,6499,1733],{"class":1091},[1050,6501,6502,6504,6506,6508,6510,6512],{"class":1052,"line":1388},[1050,6503,6467],{"class":6268},[1050,6505,1532],{"class":1091},[1050,6507,6472],{"class":1687},[1050,6509,1573],{"class":1091},[1050,6511,2179],{"class":1191},[1050,6513,1733],{"class":1091},[1050,6515,6516],{"class":1052,"line":1394},[1050,6517,2641],{"class":1067},[1050,6519,6520,6523,6525,6528,6530,6533,6535,6538,6540,6543,6545,6547],{"class":1052,"line":1777},[1050,6521,6522],{"class":1067},"        detected_type ",[1050,6524,1713],{"class":1091},[1050,6526,6527],{"class":1067}," magic",[1050,6529,1532],{"class":1091},[1050,6531,6532],{"class":1687},"from_buffer",[1050,6534,1573],{"class":1091},[1050,6536,6537],{"class":1687},"file_header",[1050,6539,1195],{"class":1091},[1050,6541,6542],{"class":1720}," mime",[1050,6544,1713],{"class":1091},[1050,6546,2100],{"class":1297},[1050,6548,1733],{"class":1091},[1050,6550,6551],{"class":1052,"line":1793},[1050,6552,2641],{"class":1067},[1050,6554,6555,6557,6559,6561,6563,6566,6569,6572,6574,6576,6578,6580],{"class":1052,"line":1809},[1050,6556,2647],{"class":1063},[1050,6558,3501],{"class":1087},[1050,6560,1532],{"class":1091},[1050,6562,6176],{"class":1649},[1050,6564,6565],{"class":1658}," and",[1050,6567,6568],{"class":1067}," detected_type ",[1050,6570,6571],{"class":1658},"not",[1050,6573,2969],{"class":1658},[1050,6575,3501],{"class":1087},[1050,6577,1532],{"class":1091},[1050,6579,6176],{"class":1649},[1050,6581,2663],{"class":1091},[1050,6583,6584,6586,6588],{"class":1052,"line":1814},[1050,6585,6390],{"class":1063},[1050,6587,6393],{"class":1687},[1050,6589,2065],{"class":1091},[1050,6591,6592,6594,6597,6599,6602,6604],{"class":1052,"line":1832},[1050,6593,6400],{"class":1565},[1050,6595,6596],{"class":1111},"'File type ",[1050,6598,3478],{"class":3477},[1050,6600,6601],{"class":1687},"detected_type",[1050,6603,3487],{"class":3477},[1050,6605,6606],{"class":1111}," is not allowed'\n",[1050,6608,6609],{"class":1052,"line":2433},[1050,6610,2535],{"class":1091},[1050,6612,6613],{"class":1052,"line":2454},[1050,6614,1598],{"class":1067},[1050,6616,6617,6619,6622,6624,6626,6628,6630],{"class":1052,"line":2470},[1050,6618,1603],{"class":1565},[1050,6620,6621],{"class":1606}," validate_filename",[1050,6623,1573],{"class":1091},[1050,6625,1613],{"class":1612},[1050,6627,1195],{"class":1091},[1050,6629,6252],{"class":1618},[1050,6631,1580],{"class":1091},[1050,6633,6634,6636,6639],{"class":1052,"line":2484},[1050,6635,1626],{"class":1585},[1050,6637,6638],{"class":1589},"Validate and sanitize filename",[1050,6640,1593],{"class":1585},[1050,6642,6643,6646,6648,6650,6652],{"class":1052,"line":2498},[1050,6644,6645],{"class":1067},"        filename ",[1050,6647,1713],{"class":1091},[1050,6649,6252],{"class":6268},[1050,6651,1532],{"class":1091},[1050,6653,6654],{"class":1649},"name\n",[1050,6656,6657],{"class":1052,"line":2511},[1050,6658,2641],{"class":1067},[1050,6660,6661],{"class":1052,"line":2522},[1050,6662,6663],{"class":1056},"        # Check for dangerous extensions\n",[1050,6665,6666,6669,6671],{"class":1052,"line":2532},[1050,6667,6668],{"class":1067},"        dangerous_extensions ",[1050,6670,1713],{"class":1091},[1050,6672,1095],{"class":1091},[1050,6674,6675,6677,6680,6682,6684,6686,6689,6691,6693,6695,6698,6700,6702,6704,6707,6709,6711,6713,6716,6718,6720,6722,6725,6727],{"class":1052,"line":2538},[1050,6676,1181],{"class":1107},[1050,6678,6679],{"class":1111},".exe",[1050,6681,1115],{"class":1107},[1050,6683,1195],{"class":1091},[1050,6685,1121],{"class":1107},[1050,6687,6688],{"class":1111},".bat",[1050,6690,1115],{"class":1107},[1050,6692,1195],{"class":1091},[1050,6694,1121],{"class":1107},[1050,6696,6697],{"class":1111},".cmd",[1050,6699,1115],{"class":1107},[1050,6701,1195],{"class":1091},[1050,6703,1121],{"class":1107},[1050,6705,6706],{"class":1111},".scr",[1050,6708,1115],{"class":1107},[1050,6710,1195],{"class":1091},[1050,6712,1121],{"class":1107},[1050,6714,6715],{"class":1111},".pif",[1050,6717,1115],{"class":1107},[1050,6719,1195],{"class":1091},[1050,6721,1121],{"class":1107},[1050,6723,6724],{"class":1111},".com",[1050,6726,1115],{"class":1107},[1050,6728,1129],{"class":1091},[1050,6730,6731,6733,6736,6738,6740,6742,6745,6747,6749,6751,6754,6756,6758,6760,6763,6765,6767,6769,6772,6774,6776,6778,6781],{"class":1052,"line":2544},[1050,6732,1181],{"class":1107},[1050,6734,6735],{"class":1111},".php",[1050,6737,1115],{"class":1107},[1050,6739,1195],{"class":1091},[1050,6741,1121],{"class":1107},[1050,6743,6744],{"class":1111},".asp",[1050,6746,1115],{"class":1107},[1050,6748,1195],{"class":1091},[1050,6750,1121],{"class":1107},[1050,6752,6753],{"class":1111},".jsp",[1050,6755,1115],{"class":1107},[1050,6757,1195],{"class":1091},[1050,6759,1121],{"class":1107},[1050,6761,6762],{"class":1111},".js",[1050,6764,1115],{"class":1107},[1050,6766,1195],{"class":1091},[1050,6768,1121],{"class":1107},[1050,6770,6771],{"class":1111},".vbs",[1050,6773,1115],{"class":1107},[1050,6775,1195],{"class":1091},[1050,6777,1121],{"class":1107},[1050,6779,6780],{"class":1111},".sh",[1050,6782,2375],{"class":1107},[1050,6784,6785],{"class":1052,"line":2549},[1050,6786,3766],{"class":1091},[1050,6788,6789],{"class":1052,"line":2554},[1050,6790,2641],{"class":1067},[1050,6792,6793,6796,6798,6800,6802,6804,6806,6809,6811,6814,6816,6819,6821,6824],{"class":1052,"line":2580},[1050,6794,6795],{"class":1067},"        file_ext ",[1050,6797,1713],{"class":1091},[1050,6799,2920],{"class":1067},[1050,6801,1532],{"class":1091},[1050,6803,3461],{"class":1649},[1050,6805,1532],{"class":1091},[1050,6807,6808],{"class":1687},"splitext",[1050,6810,1573],{"class":1091},[1050,6812,6813],{"class":1687},"filename",[1050,6815,3850],{"class":1091},[1050,6817,6818],{"class":1191},"1",[1050,6820,3855],{"class":1091},[1050,6822,6823],{"class":1687},"lower",[1050,6825,1691],{"class":1091},[1050,6827,6828,6830,6833,6835,6838],{"class":1052,"line":2585},[1050,6829,2647],{"class":1063},[1050,6831,6832],{"class":1067}," file_ext ",[1050,6834,3185],{"class":1658},[1050,6836,6837],{"class":1067}," dangerous_extensions",[1050,6839,2663],{"class":1091},[1050,6841,6842,6844,6846,6848,6850,6853,6855,6858,6860,6863],{"class":1052,"line":2603},[1050,6843,6390],{"class":1063},[1050,6845,6393],{"class":1687},[1050,6847,1573],{"class":1091},[1050,6849,3471],{"class":1565},[1050,6851,6852],{"class":1111},"'File extension ",[1050,6854,3478],{"class":3477},[1050,6856,6857],{"class":1687},"file_ext",[1050,6859,3487],{"class":3477},[1050,6861,6862],{"class":1111}," is not allowed'",[1050,6864,1733],{"class":1091},[1050,6866,6867],{"class":1052,"line":2613},[1050,6868,2641],{"class":1067},[1050,6870,6871],{"class":1052,"line":2638},[1050,6872,6873],{"class":1056},"        # Check for path traversal attempts\n",[1050,6875,6876,6878,6880,6883,6885,6887,6890,6892,6894,6896,6898,6900,6902,6904,6906,6909,6911,6913,6916],{"class":1052,"line":2644},[1050,6877,2647],{"class":1063},[1050,6879,1121],{"class":1107},[1050,6881,6882],{"class":1111},"..",[1050,6884,1115],{"class":1107},[1050,6886,2969],{"class":1658},[1050,6888,6889],{"class":1067}," filename ",[1050,6891,6184],{"class":1658},[1050,6893,1121],{"class":1107},[1050,6895,3204],{"class":1111},[1050,6897,1115],{"class":1107},[1050,6899,2969],{"class":1658},[1050,6901,6889],{"class":1067},[1050,6903,6184],{"class":1658},[1050,6905,1121],{"class":1107},[1050,6907,6908],{"class":1087},"\\\\",[1050,6910,1115],{"class":1107},[1050,6912,2969],{"class":1658},[1050,6914,6915],{"class":1067}," filename",[1050,6917,2663],{"class":1091},[1050,6919,6920,6922,6924,6926,6928,6931,6933],{"class":1052,"line":2666},[1050,6921,6390],{"class":1063},[1050,6923,6393],{"class":1687},[1050,6925,1573],{"class":1091},[1050,6927,1115],{"class":1107},[1050,6929,6930],{"class":1111},"Invalid filename",[1050,6932,1115],{"class":1107},[1050,6934,1733],{"class":1091},[1050,6936,6937],{"class":1052,"line":2672},[1050,6938,1598],{"class":1067},[1050,6940,6941,6943,6946,6948,6950,6952,6954],{"class":1052,"line":2699},[1050,6942,1603],{"class":1565},[1050,6944,6945],{"class":1606}," scan_for_malware",[1050,6947,1573],{"class":1091},[1050,6949,1613],{"class":1612},[1050,6951,1195],{"class":1091},[1050,6953,6252],{"class":1618},[1050,6955,1580],{"class":1091},[1050,6957,6958,6960,6963],{"class":1052,"line":2717},[1050,6959,1626],{"class":1585},[1050,6961,6962],{"class":1589},"Basic malware signature detection",[1050,6964,1593],{"class":1585},[1050,6966,6967,6969,6971,6973,6975,6977],{"class":1052,"line":2742},[1050,6968,6467],{"class":6268},[1050,6970,1532],{"class":1091},[1050,6972,6472],{"class":1687},[1050,6974,1573],{"class":1091},[1050,6976,2179],{"class":1191},[1050,6978,1733],{"class":1091},[1050,6980,6981,6984,6986,6988,6990,6992,6994,6997,6999],{"class":1052,"line":2748},[1050,6982,6983],{"class":1067},"        content ",[1050,6985,1713],{"class":1091},[1050,6987,6252],{"class":6268},[1050,6989,1532],{"class":1091},[1050,6991,6492],{"class":1687},[1050,6993,1573],{"class":1091},[1050,6995,6996],{"class":1191},"8192",[1050,6998,3176],{"class":1091},[1050,7000,7001],{"class":1056},"  # Read first 8KB\n",[1050,7003,7004,7006,7008,7010,7012,7014],{"class":1052,"line":2774},[1050,7005,6467],{"class":6268},[1050,7007,1532],{"class":1091},[1050,7009,6472],{"class":1687},[1050,7011,1573],{"class":1091},[1050,7013,2179],{"class":1191},[1050,7015,1733],{"class":1091},[1050,7017,7018],{"class":1052,"line":2801},[1050,7019,2641],{"class":1067},[1050,7021,7022],{"class":1052,"line":2806},[1050,7023,7024],{"class":1056},"        # Check for suspicious patterns\n",[1050,7026,7027,7030,7032],{"class":1052,"line":2814},[1050,7028,7029],{"class":1067},"        malicious_patterns ",[1050,7031,1713],{"class":1091},[1050,7033,1095],{"class":1091},[1050,7035,7036,7039,7041,7044,7046],{"class":1052,"line":2819},[1050,7037,7038],{"class":1565},"            b",[1050,7040,1115],{"class":1107},[1050,7042,7043],{"class":1111},"\u003Cscript",[1050,7045,1115],{"class":1107},[1050,7047,1129],{"class":1091},[1050,7049,7050,7052,7054,7057,7059],{"class":1052,"line":4159},[1050,7051,7038],{"class":1565},[1050,7053,1115],{"class":1107},[1050,7055,7056],{"class":1111},"javascript:",[1050,7058,1115],{"class":1107},[1050,7060,1129],{"class":1091},[1050,7062,7063,7065,7067,7070,7072],{"class":1052,"line":4168},[1050,7064,7038],{"class":1565},[1050,7066,1115],{"class":1107},[1050,7068,7069],{"class":1111},"\u003C?php",[1050,7071,1115],{"class":1107},[1050,7073,1129],{"class":1091},[1050,7075,7076,7078,7080,7083,7085],{"class":1052,"line":4180},[1050,7077,7038],{"class":1565},[1050,7079,1115],{"class":1107},[1050,7081,7082],{"class":1111},"\u003C%",[1050,7084,1115],{"class":1107},[1050,7086,1129],{"class":1091},[1050,7088,7089,7091,7093,7096,7098],{"class":1052,"line":4185},[1050,7090,7038],{"class":1565},[1050,7092,1115],{"class":1107},[1050,7094,7095],{"class":1111},"eval(",[1050,7097,1115],{"class":1107},[1050,7099,1129],{"class":1091},[1050,7101,7102,7104,7106,7109,7111],{"class":1052,"line":4190},[1050,7103,7038],{"class":1565},[1050,7105,1115],{"class":1107},[1050,7107,7108],{"class":1111},"exec(",[1050,7110,1115],{"class":1107},[1050,7112,1129],{"class":1091},[1050,7114,7115,7117,7119,7122,7124],{"class":1052,"line":4195},[1050,7116,7038],{"class":1565},[1050,7118,1115],{"class":1107},[1050,7120,7121],{"class":1111},"system(",[1050,7123,1115],{"class":1107},[1050,7125,1129],{"class":1091},[1050,7127,7128,7130,7132,7135,7137],{"class":1052,"line":4201},[1050,7129,7038],{"class":1565},[1050,7131,1115],{"class":1107},[1050,7133,7134],{"class":1111},"shell_exec(",[1050,7136,1115],{"class":1107},[1050,7138,1129],{"class":1091},[1050,7140,7141],{"class":1052,"line":4210},[1050,7142,3766],{"class":1091},[1050,7144,7145],{"class":1052,"line":4225},[1050,7146,2641],{"class":1067},[1050,7148,7149,7152,7154,7156,7158,7160],{"class":1052,"line":4240},[1050,7150,7151],{"class":1067},"        content_lower ",[1050,7153,1713],{"class":1091},[1050,7155,5527],{"class":1067},[1050,7157,1532],{"class":1091},[1050,7159,6823],{"class":1687},[1050,7161,1691],{"class":1091},[1050,7163,7164,7166,7169,7171,7174],{"class":1052,"line":4255},[1050,7165,3775],{"class":1063},[1050,7167,7168],{"class":1067}," pattern ",[1050,7170,3185],{"class":1063},[1050,7172,7173],{"class":1067}," malicious_patterns",[1050,7175,2663],{"class":1091},[1050,7177,7179,7181,7183,7185,7188],{"class":1052,"line":7178},81,[1050,7180,2675],{"class":1063},[1050,7182,7168],{"class":1067},[1050,7184,3185],{"class":1658},[1050,7186,7187],{"class":1067}," content_lower",[1050,7189,2663],{"class":1091},[1050,7191,7193,7196,7198,7200,7202,7205,7207],{"class":1052,"line":7192},82,[1050,7194,7195],{"class":1063},"                raise",[1050,7197,6393],{"class":1687},[1050,7199,1573],{"class":1091},[1050,7201,1115],{"class":1107},[1050,7203,7204],{"class":1111},"Potentially malicious content detected",[1050,7206,1115],{"class":1107},[1050,7208,1733],{"class":1091},[1050,7210,7212],{"class":1052,"line":7211},83,[1050,7213,1075],{"emptyLinePlaceholder":1074},[1050,7215,7217],{"class":1052,"line":7216},84,[1050,7218,7219],{"class":1056},"# Usage in admin\n",[1050,7221,7223,7226,7228,7230],{"class":1052,"line":7222},85,[1050,7224,7225],{"class":1067},"secure_image_validator ",[1050,7227,1713],{"class":1091},[1050,7229,6112],{"class":1687},[1050,7231,2065],{"class":1091},[1050,7233,7235,7238,7240,7242,7245,7247,7249,7251,7254,7256,7258,7260,7263,7265],{"class":1052,"line":7234},86,[1050,7236,7237],{"class":1720},"    allowed_types",[1050,7239,2723],{"class":1091},[1050,7241,1115],{"class":1107},[1050,7243,7244],{"class":1111},"image/jpeg",[1050,7246,1115],{"class":1107},[1050,7248,1195],{"class":1091},[1050,7250,1121],{"class":1107},[1050,7252,7253],{"class":1111},"image/png",[1050,7255,1115],{"class":1107},[1050,7257,1195],{"class":1091},[1050,7259,1121],{"class":1107},[1050,7261,7262],{"class":1111},"image/gif",[1050,7264,1115],{"class":1107},[1050,7266,7267],{"class":1091},"],\n",[1050,7269,7271,7274,7276,7279,7281,7283,7285,7287],{"class":1052,"line":7270},87,[1050,7272,7273],{"class":1720},"    max_size",[1050,7275,1713],{"class":1091},[1050,7277,7278],{"class":1191},"5",[1050,7280,6209],{"class":1658},[1050,7282,6212],{"class":1191},[1050,7284,6209],{"class":1658},[1050,7286,6212],{"class":1191},[1050,7288,7289],{"class":1056},"  # 5MB\n",[1050,7291,7293],{"class":1052,"line":7292},88,[1050,7294,1733],{"class":1091},[1050,7296,7298],{"class":1052,"line":7297},89,[1050,7299,1075],{"emptyLinePlaceholder":1074},[1050,7301,7303,7305,7308,7310,7312,7314,7316],{"class":1052,"line":7302},90,[1050,7304,1566],{"class":1565},[1050,7306,7307],{"class":1569}," MediaFileAdmin",[1050,7309,1573],{"class":1091},[1050,7311,2822],{"class":1576},[1050,7313,1532],{"class":1091},[1050,7315,5396],{"class":1576},[1050,7317,1580],{"class":1091},[1050,7319,7321,7323,7326,7328,7330,7332,7335,7337,7340,7343],{"class":1052,"line":7320},91,[1050,7322,1603],{"class":1565},[1050,7324,7325],{"class":1606}," formfield_for_dbfield",[1050,7327,1573],{"class":1091},[1050,7329,1613],{"class":1612},[1050,7331,1195],{"class":1091},[1050,7333,7334],{"class":1618}," db_field",[1050,7336,1195],{"class":1091},[1050,7338,7339],{"class":1658}," **",[1050,7341,7342],{"class":1618},"kwargs",[1050,7344,1580],{"class":1091},[1050,7346,7348,7350,7353],{"class":1052,"line":7347},92,[1050,7349,1626],{"class":1585},[1050,7351,7352],{"class":1589},"Add validators to file fields",[1050,7354,1593],{"class":1585},[1050,7356,7358,7360,7362,7364,7366,7368,7370,7372,7375,7377,7379,7381,7383,7385],{"class":1052,"line":7357},93,[1050,7359,2647],{"class":1063},[1050,7361,7334],{"class":1067},[1050,7363,1532],{"class":1091},[1050,7365,1721],{"class":1649},[1050,7367,2969],{"class":1658},[1050,7369,2393],{"class":1091},[1050,7371,1115],{"class":1107},[1050,7373,7374],{"class":1111},"image",[1050,7376,1115],{"class":1107},[1050,7378,1195],{"class":1091},[1050,7380,1121],{"class":1107},[1050,7382,6269],{"class":1111},[1050,7384,1115],{"class":1107},[1050,7386,3007],{"class":1091},[1050,7388,7390,7393,7396,7398,7401,7403,7405,7407,7409,7412],{"class":1052,"line":7389},94,[1050,7391,7392],{"class":1067},"            kwargs",[1050,7394,7395],{"class":1091},"[",[1050,7397,1115],{"class":1107},[1050,7399,7400],{"class":1111},"validators",[1050,7402,1115],{"class":1107},[1050,7404,6039],{"class":1091},[1050,7406,1092],{"class":1091},[1050,7408,2393],{"class":1091},[1050,7410,7411],{"class":1067},"secure_image_validator",[1050,7413,1275],{"class":1091},[1050,7415,7417],{"class":1052,"line":7416},95,[1050,7418,2641],{"class":1067},[1050,7420,7422,7424,7426,7428,7431,7433,7436,7438,7440,7442],{"class":1052,"line":7421},96,[1050,7423,1636],{"class":1063},[1050,7425,2622],{"class":2621},[1050,7427,2625],{"class":1091},[1050,7429,7430],{"class":1687},"formfield_for_dbfield",[1050,7432,1573],{"class":1091},[1050,7434,7435],{"class":1687},"db_field",[1050,7437,1195],{"class":1091},[1050,7439,7339],{"class":1658},[1050,7441,7342],{"class":1687},[1050,7443,1733],{"class":1091},[1030,7445,767],{"id":7446},"monitoring-and-logging",[1035,7448,7450],{"id":7449},"comprehensive-admin-logging","Comprehensive Admin Logging",[1040,7452,7454],{"className":1042,"code":7453,"language":1044,"meta":1045,"style":1045},"# logging_middleware.py\nimport logging\nimport json\nfrom django.utils import timezone\n\nlogger = logging.getLogger('admin_security')\n\nclass AdminSecurityMiddleware:\n    \"\"\"Log all admin activities for security monitoring\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n        self.admin_url = getattr(settings, 'ADMIN_URL', 'admin/')\n    \n    def __call__(self, request):\n        if request.path.startswith(f'/{self.admin_url}'):\n            self.log_admin_access(request)\n        \n        response = self.get_response(request)\n        \n        if request.path.startswith(f'/{self.admin_url}'):\n            self.log_admin_response(request, response)\n        \n        return response\n    \n    def log_admin_access(self, request):\n        \"\"\"Log admin access attempts\"\"\"\n        log_data = {\n            'timestamp': timezone.now().isoformat(),\n            'ip_address': self.get_client_ip(request),\n            'user_agent': request.META.get('HTTP_USER_AGENT', ''),\n            'method': request.method,\n            'path': request.path,\n            'user': getattr(request.user, 'username', 'Anonymous'),\n            'is_authenticated': request.user.is_authenticated,\n            'session_key': request.session.session_key,\n        }\n        \n        # Log POST data (excluding sensitive fields)\n        if request.method == 'POST':\n            post_data = dict(request.POST)\n            # Remove sensitive fields\n            sensitive_fields = ['password', 'csrfmiddlewaretoken']\n            for field in sensitive_fields:\n                post_data.pop(field, None)\n            log_data['post_data'] = post_data\n        \n        logger.info(f'Admin access: {json.dumps(log_data)}')\n    \n    def log_admin_response(self, request, response):\n        \"\"\"Log admin response status\"\"\"\n        if response.status_code >= 400:\n            logger.warning(\n                f'Admin error response: {response.status_code} - '\n                f'User: {getattr(request.user, \"username\", \"Anonymous\")} - '\n                f'Path: {request.path} - '\n                f'IP: {self.get_client_ip(request)}'\n            )\n\n# settings.py\nLOGGING = {\n    'version': 1,\n    'disable_existing_loggers': False,\n    'formatters': {\n        'verbose': {\n            'format': '{levelname} {asctime} {module} {process:d} {thread:d} {message}',\n            'style': '{',\n        },\n        'security': {\n            'format': '{asctime} SECURITY {levelname} {message}',\n            'style': '{',\n        },\n    },\n    'handlers': {\n        'admin_security_file': {\n            'level': 'INFO',\n            'class': 'logging.handlers.RotatingFileHandler',\n            'filename': '/var/log/django/admin_security.log',\n            'maxBytes': 10 * 1024 * 1024,  # 10MB\n            'backupCount': 5,\n            'formatter': 'security',\n        },\n        'security_email': {\n            'level': 'WARNING',\n            'class': 'django.utils.log.AdminEmailHandler',\n            'formatter': 'verbose',\n        },\n    },\n    'loggers': {\n        'admin_security': {\n            'handlers': ['admin_security_file', 'security_email'],\n            'level': 'INFO',\n            'propagate': False,\n        },\n    },\n}\n",[1047,7455,7456,7461,7467,7474,7488,7492,7515,7519,7528,7537,7541,7557,7569,7603,7607,7623,7657,7672,7676,7695,7699,7733,7753,7757,7764,7768,7785,7794,7803,7828,7850,7886,7904,7922,7960,7983,8007,8011,8015,8020,8040,8060,8065,8091,8105,8127,8148,8152,8190,8194,8215,8224,8243,8253,8273,8313,8331,8355,8359,8363,8367,8376,8392,8408,8421,8434,8478,8497,8502,8515,8541,8559,8563,8567,8580,8593,8613,8632,8651,8677,8693,8712,8716,8729,8748,8767,8785,8789,8793,8806,8818,8846,8864,8879,8883,8887],{"__ignoreMap":1045},[1050,7457,7458],{"class":1052,"line":1053},[1050,7459,7460],{"class":1056},"# logging_middleware.py\n",[1050,7462,7463,7465],{"class":1052,"line":1060},[1050,7464,1064],{"class":1063},[1050,7466,3275],{"class":1067},[1050,7468,7469,7471],{"class":1052,"line":1071},[1050,7470,1064],{"class":1063},[1050,7472,7473],{"class":1067}," json\n",[1050,7475,7476,7478,7480,7482,7484,7486],{"class":1052,"line":1078},[1050,7477,1526],{"class":1063},[1050,7479,1749],{"class":1067},[1050,7481,1532],{"class":1091},[1050,7483,2271],{"class":1067},[1050,7485,1064],{"class":1063},[1050,7487,2276],{"class":1067},[1050,7489,7490],{"class":1052,"line":1084},[1050,7491,1075],{"emptyLinePlaceholder":1074},[1050,7493,7494,7496,7498,7500,7502,7504,7506,7508,7511,7513],{"class":1052,"line":1098},[1050,7495,3284],{"class":1067},[1050,7497,1713],{"class":1091},[1050,7499,3289],{"class":1067},[1050,7501,1532],{"class":1091},[1050,7503,3294],{"class":1687},[1050,7505,1573],{"class":1091},[1050,7507,1115],{"class":1107},[1050,7509,7510],{"class":1111},"admin_security",[1050,7512,1115],{"class":1107},[1050,7514,1733],{"class":1091},[1050,7516,7517],{"class":1052,"line":1104},[1050,7518,1075],{"emptyLinePlaceholder":1074},[1050,7520,7521,7523,7526],{"class":1052,"line":1132},[1050,7522,1566],{"class":1565},[1050,7524,7525],{"class":1569}," AdminSecurityMiddleware",[1050,7527,2663],{"class":1091},[1050,7529,7530,7532,7535],{"class":1052,"line":1138},[1050,7531,1586],{"class":1585},[1050,7533,7534],{"class":1589},"Log all admin activities for security monitoring",[1050,7536,1593],{"class":1585},[1050,7538,7539],{"class":1052,"line":1143},[1050,7540,1598],{"class":1067},[1050,7542,7543,7545,7547,7549,7551,7553,7555],{"class":1052,"line":1163},[1050,7544,1603],{"class":1565},[1050,7546,3335],{"class":3188},[1050,7548,1573],{"class":1091},[1050,7550,1613],{"class":1612},[1050,7552,1195],{"class":1091},[1050,7554,3344],{"class":1618},[1050,7556,1580],{"class":1091},[1050,7558,7559,7561,7563,7565,7567],{"class":1052,"line":1178},[1050,7560,3351],{"class":1087},[1050,7562,1532],{"class":1091},[1050,7564,3356],{"class":1649},[1050,7566,1092],{"class":1091},[1050,7568,3361],{"class":1067},[1050,7570,7571,7573,7575,7577,7579,7581,7583,7585,7587,7589,7591,7593,7595,7597,7599,7601],{"class":1052,"line":1201},[1050,7572,3351],{"class":1087},[1050,7574,1532],{"class":1091},[1050,7576,3403],{"class":1649},[1050,7578,1092],{"class":1091},[1050,7580,3375],{"class":3188},[1050,7582,1573],{"class":1091},[1050,7584,3380],{"class":1687},[1050,7586,1195],{"class":1091},[1050,7588,1121],{"class":1107},[1050,7590,2915],{"class":1111},[1050,7592,1115],{"class":1107},[1050,7594,1195],{"class":1091},[1050,7596,1121],{"class":1107},[1050,7598,2945],{"class":1111},[1050,7600,1115],{"class":1107},[1050,7602,1733],{"class":1091},[1050,7604,7605],{"class":1052,"line":1207},[1050,7606,1598],{"class":1067},[1050,7608,7609,7611,7613,7615,7617,7619,7621],{"class":1052,"line":1212},[1050,7610,1603],{"class":1565},[1050,7612,3440],{"class":3188},[1050,7614,1573],{"class":1091},[1050,7616,1613],{"class":1612},[1050,7618,1195],{"class":1091},[1050,7620,1619],{"class":1618},[1050,7622,1580],{"class":1091},[1050,7624,7625,7627,7629,7631,7633,7635,7637,7639,7641,7643,7645,7647,7649,7651,7653,7655],{"class":1052,"line":1217},[1050,7626,2647],{"class":1063},[1050,7628,1619],{"class":1067},[1050,7630,1532],{"class":1091},[1050,7632,3461],{"class":1649},[1050,7634,1532],{"class":1091},[1050,7636,3466],{"class":1687},[1050,7638,1573],{"class":1091},[1050,7640,3471],{"class":1565},[1050,7642,3474],{"class":1111},[1050,7644,3478],{"class":3477},[1050,7646,1613],{"class":1087},[1050,7648,1532],{"class":1091},[1050,7650,3403],{"class":1649},[1050,7652,3487],{"class":3477},[1050,7654,1115],{"class":1111},[1050,7656,1580],{"class":1091},[1050,7658,7659,7661,7663,7666,7668,7670],{"class":1052,"line":1237},[1050,7660,6322],{"class":1087},[1050,7662,1532],{"class":1091},[1050,7664,7665],{"class":1687},"log_admin_access",[1050,7667,1573],{"class":1091},[1050,7669,2633],{"class":1687},[1050,7671,1733],{"class":1091},[1050,7673,7674],{"class":1052,"line":1242},[1050,7675,2641],{"class":1067},[1050,7677,7678,7681,7683,7685,7687,7689,7691,7693],{"class":1052,"line":1247},[1050,7679,7680],{"class":1067},"        response ",[1050,7682,1713],{"class":1091},[1050,7684,3501],{"class":1087},[1050,7686,1532],{"class":1091},[1050,7688,3356],{"class":1687},[1050,7690,1573],{"class":1091},[1050,7692,2633],{"class":1687},[1050,7694,1733],{"class":1091},[1050,7696,7697],{"class":1052,"line":1267},[1050,7698,2641],{"class":1067},[1050,7700,7701,7703,7705,7707,7709,7711,7713,7715,7717,7719,7721,7723,7725,7727,7729,7731],{"class":1052,"line":1272},[1050,7702,2647],{"class":1063},[1050,7704,1619],{"class":1067},[1050,7706,1532],{"class":1091},[1050,7708,3461],{"class":1649},[1050,7710,1532],{"class":1091},[1050,7712,3466],{"class":1687},[1050,7714,1573],{"class":1091},[1050,7716,3471],{"class":1565},[1050,7718,3474],{"class":1111},[1050,7720,3478],{"class":3477},[1050,7722,1613],{"class":1087},[1050,7724,1532],{"class":1091},[1050,7726,3403],{"class":1649},[1050,7728,3487],{"class":3477},[1050,7730,1115],{"class":1111},[1050,7732,1580],{"class":1091},[1050,7734,7735,7737,7739,7742,7744,7746,7748,7751],{"class":1052,"line":1278},[1050,7736,6322],{"class":1087},[1050,7738,1532],{"class":1091},[1050,7740,7741],{"class":1687},"log_admin_response",[1050,7743,1573],{"class":1091},[1050,7745,2633],{"class":1687},[1050,7747,1195],{"class":1091},[1050,7749,7750],{"class":1687}," response",[1050,7752,1733],{"class":1091},[1050,7754,7755],{"class":1052,"line":1283},[1050,7756,2641],{"class":1067},[1050,7758,7759,7761],{"class":1052,"line":1289},[1050,7760,1636],{"class":1063},[1050,7762,7763],{"class":1067}," response\n",[1050,7765,7766],{"class":1052,"line":1304},[1050,7767,1598],{"class":1067},[1050,7769,7770,7772,7775,7777,7779,7781,7783],{"class":1052,"line":1317},[1050,7771,1603],{"class":1565},[1050,7773,7774],{"class":1606}," log_admin_access",[1050,7776,1573],{"class":1091},[1050,7778,1613],{"class":1612},[1050,7780,1195],{"class":1091},[1050,7782,1619],{"class":1618},[1050,7784,1580],{"class":1091},[1050,7786,7787,7789,7792],{"class":1052,"line":1331},[1050,7788,1626],{"class":1585},[1050,7790,7791],{"class":1589},"Log admin access attempts",[1050,7793,1593],{"class":1585},[1050,7795,7796,7799,7801],{"class":1052,"line":1342},[1050,7797,7798],{"class":1067},"        log_data ",[1050,7800,1713],{"class":1091},[1050,7802,1175],{"class":1091},[1050,7804,7805,7807,7810,7812,7814,7816,7818,7820,7822,7825],{"class":1052,"line":1347},[1050,7806,1181],{"class":1107},[1050,7808,7809],{"class":1111},"timestamp",[1050,7811,1115],{"class":1107},[1050,7813,1118],{"class":1091},[1050,7815,5233],{"class":1067},[1050,7817,1532],{"class":1091},[1050,7819,5238],{"class":1687},[1050,7821,2625],{"class":1091},[1050,7823,7824],{"class":1687},"isoformat",[1050,7826,7827],{"class":1091},"(),\n",[1050,7829,7830,7832,7834,7836,7838,7840,7842,7844,7846,7848],{"class":1052,"line":1353},[1050,7831,1181],{"class":1107},[1050,7833,3877],{"class":1111},[1050,7835,1115],{"class":1107},[1050,7837,1118],{"class":1091},[1050,7839,3501],{"class":1087},[1050,7841,1532],{"class":1091},[1050,7843,3506],{"class":1687},[1050,7845,1573],{"class":1091},[1050,7847,2633],{"class":1687},[1050,7849,1972],{"class":1091},[1050,7851,7852,7854,7857,7859,7861,7863,7865,7867,7869,7871,7873,7875,7878,7880,7882,7884],{"class":1052,"line":1363},[1050,7853,1181],{"class":1107},[1050,7855,7856],{"class":1111},"user_agent",[1050,7858,1115],{"class":1107},[1050,7860,1118],{"class":1091},[1050,7862,1619],{"class":1067},[1050,7864,1532],{"class":1091},[1050,7866,3800],{"class":3799},[1050,7868,1532],{"class":1091},[1050,7870,2930],{"class":1687},[1050,7872,1573],{"class":1091},[1050,7874,1115],{"class":1107},[1050,7876,7877],{"class":1111},"HTTP_USER_AGENT",[1050,7879,1115],{"class":1107},[1050,7881,1195],{"class":1091},[1050,7883,3153],{"class":1107},[1050,7885,1972],{"class":1091},[1050,7887,7888,7890,7892,7894,7896,7898,7900,7902],{"class":1052,"line":1373},[1050,7889,1181],{"class":1107},[1050,7891,4571],{"class":1111},[1050,7893,1115],{"class":1107},[1050,7895,1118],{"class":1091},[1050,7897,1619],{"class":1067},[1050,7899,1532],{"class":1091},[1050,7901,4571],{"class":1649},[1050,7903,1129],{"class":1091},[1050,7905,7906,7908,7910,7912,7914,7916,7918,7920],{"class":1052,"line":1383},[1050,7907,1181],{"class":1107},[1050,7909,3461],{"class":1111},[1050,7911,1115],{"class":1107},[1050,7913,1118],{"class":1091},[1050,7915,1619],{"class":1067},[1050,7917,1532],{"class":1091},[1050,7919,3461],{"class":1649},[1050,7921,1129],{"class":1091},[1050,7923,7924,7926,7928,7930,7932,7934,7936,7938,7940,7942,7944,7946,7948,7950,7952,7954,7956,7958],{"class":1052,"line":1388},[1050,7925,1181],{"class":1107},[1050,7927,1650],{"class":1111},[1050,7929,1115],{"class":1107},[1050,7931,1118],{"class":1091},[1050,7933,3375],{"class":3188},[1050,7935,1573],{"class":1091},[1050,7937,2633],{"class":1687},[1050,7939,1532],{"class":1091},[1050,7941,1650],{"class":1649},[1050,7943,1195],{"class":1091},[1050,7945,1121],{"class":1107},[1050,7947,2323],{"class":1111},[1050,7949,1115],{"class":1107},[1050,7951,1195],{"class":1091},[1050,7953,1121],{"class":1107},[1050,7955,3602],{"class":1111},[1050,7957,1115],{"class":1107},[1050,7959,1972],{"class":1091},[1050,7961,7962,7964,7967,7969,7971,7973,7975,7977,7979,7981],{"class":1052,"line":1394},[1050,7963,1181],{"class":1107},[1050,7965,7966],{"class":1111},"is_authenticated",[1050,7968,1115],{"class":1107},[1050,7970,1118],{"class":1091},[1050,7972,1619],{"class":1067},[1050,7974,1532],{"class":1091},[1050,7976,1650],{"class":1649},[1050,7978,1532],{"class":1091},[1050,7980,7966],{"class":1649},[1050,7982,1129],{"class":1091},[1050,7984,7985,7987,7990,7992,7994,7996,7998,8001,8003,8005],{"class":1052,"line":1777},[1050,7986,1181],{"class":1107},[1050,7988,7989],{"class":1111},"session_key",[1050,7991,1115],{"class":1107},[1050,7993,1118],{"class":1091},[1050,7995,1619],{"class":1067},[1050,7997,1532],{"class":1091},[1050,7999,8000],{"class":1649},"session",[1050,8002,1532],{"class":1091},[1050,8004,7989],{"class":1649},[1050,8006,1129],{"class":1091},[1050,8008,8009],{"class":1052,"line":1793},[1050,8010,1204],{"class":1091},[1050,8012,8013],{"class":1052,"line":1809},[1050,8014,2641],{"class":1067},[1050,8016,8017],{"class":1052,"line":1814},[1050,8018,8019],{"class":1056},"        # Log POST data (excluding sensitive fields)\n",[1050,8021,8022,8024,8026,8028,8030,8032,8034,8036,8038],{"class":1052,"line":1832},[1050,8023,2647],{"class":1063},[1050,8025,1619],{"class":1067},[1050,8027,1532],{"class":1091},[1050,8029,4571],{"class":1649},[1050,8031,2688],{"class":1658},[1050,8033,1121],{"class":1107},[1050,8035,4578],{"class":1111},[1050,8037,1115],{"class":1107},[1050,8039,2663],{"class":1091},[1050,8041,8042,8045,8047,8050,8052,8054,8056,8058],{"class":1052,"line":2433},[1050,8043,8044],{"class":1067},"            post_data ",[1050,8046,1713],{"class":1091},[1050,8048,8049],{"class":2621}," dict",[1050,8051,1573],{"class":1091},[1050,8053,2633],{"class":1687},[1050,8055,1532],{"class":1091},[1050,8057,4578],{"class":3799},[1050,8059,1733],{"class":1091},[1050,8061,8062],{"class":1052,"line":2454},[1050,8063,8064],{"class":1056},"            # Remove sensitive fields\n",[1050,8066,8067,8070,8072,8074,8076,8078,8080,8082,8084,8087,8089],{"class":1052,"line":2470},[1050,8068,8069],{"class":1067},"            sensitive_fields ",[1050,8071,1713],{"class":1091},[1050,8073,2393],{"class":1091},[1050,8075,1115],{"class":1107},[1050,8077,4633],{"class":1111},[1050,8079,1115],{"class":1107},[1050,8081,1195],{"class":1091},[1050,8083,1121],{"class":1107},[1050,8085,8086],{"class":1111},"csrfmiddlewaretoken",[1050,8088,1115],{"class":1107},[1050,8090,1275],{"class":1091},[1050,8092,8093,8095,8098,8100,8103],{"class":1052,"line":2484},[1050,8094,4032],{"class":1063},[1050,8096,8097],{"class":1067}," field ",[1050,8099,3185],{"class":1063},[1050,8101,8102],{"class":1067}," sensitive_fields",[1050,8104,2663],{"class":1091},[1050,8106,8107,8110,8112,8115,8117,8120,8122,8125],{"class":1052,"line":2498},[1050,8108,8109],{"class":1067},"                post_data",[1050,8111,1532],{"class":1091},[1050,8113,8114],{"class":1687},"pop",[1050,8116,1573],{"class":1091},[1050,8118,8119],{"class":1687},"field",[1050,8121,1195],{"class":1091},[1050,8123,8124],{"class":1297}," None",[1050,8126,1733],{"class":1091},[1050,8128,8129,8132,8134,8136,8139,8141,8143,8145],{"class":1052,"line":2511},[1050,8130,8131],{"class":1067},"            log_data",[1050,8133,7395],{"class":1091},[1050,8135,1115],{"class":1107},[1050,8137,8138],{"class":1111},"post_data",[1050,8140,1115],{"class":1107},[1050,8142,6039],{"class":1091},[1050,8144,1092],{"class":1091},[1050,8146,8147],{"class":1067}," post_data\n",[1050,8149,8150],{"class":1052,"line":2522},[1050,8151,2641],{"class":1067},[1050,8153,8154,8156,8158,8160,8162,8164,8167,8169,8172,8174,8177,8179,8182,8184,8186,8188],{"class":1052,"line":2532},[1050,8155,5057],{"class":1067},[1050,8157,1532],{"class":1091},[1050,8159,5872],{"class":1687},[1050,8161,1573],{"class":1091},[1050,8163,3471],{"class":1565},[1050,8165,8166],{"class":1111},"'Admin access: ",[1050,8168,3478],{"class":3477},[1050,8170,8171],{"class":1687},"json",[1050,8173,1532],{"class":1091},[1050,8175,8176],{"class":1687},"dumps",[1050,8178,1573],{"class":1091},[1050,8180,8181],{"class":1687},"log_data",[1050,8183,3176],{"class":1091},[1050,8185,3487],{"class":3477},[1050,8187,1115],{"class":1111},[1050,8189,1733],{"class":1091},[1050,8191,8192],{"class":1052,"line":2538},[1050,8193,1598],{"class":1067},[1050,8195,8196,8198,8201,8203,8205,8207,8209,8211,8213],{"class":1052,"line":2544},[1050,8197,1603],{"class":1565},[1050,8199,8200],{"class":1606}," log_admin_response",[1050,8202,1573],{"class":1091},[1050,8204,1613],{"class":1612},[1050,8206,1195],{"class":1091},[1050,8208,1619],{"class":1618},[1050,8210,1195],{"class":1091},[1050,8212,7750],{"class":1618},[1050,8214,1580],{"class":1091},[1050,8216,8217,8219,8222],{"class":1052,"line":2549},[1050,8218,1626],{"class":1585},[1050,8220,8221],{"class":1589},"Log admin response status",[1050,8223,1593],{"class":1585},[1050,8225,8226,8228,8230,8232,8235,8238,8241],{"class":1052,"line":2554},[1050,8227,2647],{"class":1063},[1050,8229,7750],{"class":1067},[1050,8231,1532],{"class":1091},[1050,8233,8234],{"class":1649},"status_code",[1050,8236,8237],{"class":1658}," >=",[1050,8239,8240],{"class":1191}," 400",[1050,8242,2663],{"class":1091},[1050,8244,8245,8247,8249,8251],{"class":1052,"line":2580},[1050,8246,4105],{"class":1067},[1050,8248,1532],{"class":1091},[1050,8250,3547],{"class":1687},[1050,8252,2065],{"class":1091},[1050,8254,8255,8257,8260,8262,8265,8267,8269,8271],{"class":1052,"line":2585},[1050,8256,6400],{"class":1565},[1050,8258,8259],{"class":1111},"'Admin error response: ",[1050,8261,3478],{"class":3477},[1050,8263,8264],{"class":1687},"response",[1050,8266,1532],{"class":1091},[1050,8268,8234],{"class":1649},[1050,8270,3487],{"class":3477},[1050,8272,3566],{"class":1111},[1050,8274,8275,8277,8279,8281,8283,8285,8287,8289,8291,8293,8295,8297,8299,8301,8303,8305,8307,8309,8311],{"class":1052,"line":2603},[1050,8276,6400],{"class":1565},[1050,8278,3573],{"class":1111},[1050,8280,3478],{"class":3477},[1050,8282,3578],{"class":3188},[1050,8284,1573],{"class":1091},[1050,8286,2633],{"class":1687},[1050,8288,1532],{"class":1091},[1050,8290,1650],{"class":1649},[1050,8292,1195],{"class":1091},[1050,8294,3591],{"class":1107},[1050,8296,2323],{"class":1111},[1050,8298,3020],{"class":1107},[1050,8300,1195],{"class":1091},[1050,8302,3591],{"class":1107},[1050,8304,3602],{"class":1111},[1050,8306,3020],{"class":1107},[1050,8308,3176],{"class":1091},[1050,8310,3487],{"class":3477},[1050,8312,3566],{"class":1111},[1050,8314,8315,8317,8319,8321,8323,8325,8327,8329],{"class":1052,"line":2613},[1050,8316,6400],{"class":1565},[1050,8318,3617],{"class":1111},[1050,8320,3478],{"class":3477},[1050,8322,2633],{"class":1687},[1050,8324,1532],{"class":1091},[1050,8326,3461],{"class":1649},[1050,8328,3487],{"class":3477},[1050,8330,3566],{"class":1111},[1050,8332,8333,8335,8337,8339,8341,8343,8345,8347,8349,8351,8353],{"class":1052,"line":2638},[1050,8334,6400],{"class":1565},[1050,8336,5087],{"class":1111},[1050,8338,3478],{"class":3477},[1050,8340,1613],{"class":1087},[1050,8342,1532],{"class":1091},[1050,8344,3506],{"class":1687},[1050,8346,1573],{"class":1091},[1050,8348,2633],{"class":1687},[1050,8350,3176],{"class":1091},[1050,8352,3487],{"class":3477},[1050,8354,2375],{"class":1111},[1050,8356,8357],{"class":1052,"line":2644},[1050,8358,2535],{"class":1091},[1050,8360,8361],{"class":1052,"line":2666},[1050,8362,1075],{"emptyLinePlaceholder":1074},[1050,8364,8365],{"class":1052,"line":2672},[1050,8366,1057],{"class":1056},[1050,8368,8369,8372,8374],{"class":1052,"line":2699},[1050,8370,8371],{"class":1087},"LOGGING",[1050,8373,1092],{"class":1091},[1050,8375,1175],{"class":1091},[1050,8377,8378,8380,8383,8385,8387,8390],{"class":1052,"line":2717},[1050,8379,1441],{"class":1107},[1050,8381,8382],{"class":1111},"version",[1050,8384,1115],{"class":1107},[1050,8386,1118],{"class":1091},[1050,8388,8389],{"class":1191}," 1",[1050,8391,1129],{"class":1091},[1050,8393,8394,8396,8399,8401,8403,8406],{"class":1052,"line":2742},[1050,8395,1441],{"class":1107},[1050,8397,8398],{"class":1111},"disable_existing_loggers",[1050,8400,1115],{"class":1107},[1050,8402,1118],{"class":1091},[1050,8404,8405],{"class":1297}," False",[1050,8407,1129],{"class":1091},[1050,8409,8410,8412,8415,8417,8419],{"class":1052,"line":2748},[1050,8411,1441],{"class":1107},[1050,8413,8414],{"class":1111},"formatters",[1050,8416,1115],{"class":1107},[1050,8418,1118],{"class":1091},[1050,8420,1175],{"class":1091},[1050,8422,8423,8425,8428,8430,8432],{"class":1052,"line":2774},[1050,8424,1108],{"class":1107},[1050,8426,8427],{"class":1111},"verbose",[1050,8429,1115],{"class":1107},[1050,8431,1118],{"class":1091},[1050,8433,1175],{"class":1091},[1050,8435,8436,8438,8441,8443,8445,8447,8450,8453,8456,8459,8462,8464,8467,8469,8471,8474,8476],{"class":1052,"line":2801},[1050,8437,1181],{"class":1107},[1050,8439,8440],{"class":1111},"format",[1050,8442,1115],{"class":1107},[1050,8444,1118],{"class":1091},[1050,8446,1121],{"class":1107},[1050,8448,8449],{"class":3477},"{levelname}",[1050,8451,8452],{"class":3477}," {asctime}",[1050,8454,8455],{"class":3477}," {module}",[1050,8457,8458],{"class":3477}," {process",[1050,8460,8461],{"class":1565},":d",[1050,8463,3487],{"class":3477},[1050,8465,8466],{"class":3477}," {thread",[1050,8468,8461],{"class":1565},[1050,8470,3487],{"class":3477},[1050,8472,8473],{"class":3477}," {message}",[1050,8475,1115],{"class":1107},[1050,8477,1129],{"class":1091},[1050,8479,8480,8482,8485,8487,8489,8491,8493,8495],{"class":1052,"line":2806},[1050,8481,1181],{"class":1107},[1050,8483,8484],{"class":1111},"style",[1050,8486,1115],{"class":1107},[1050,8488,1118],{"class":1091},[1050,8490,1121],{"class":1107},[1050,8492,3478],{"class":1111},[1050,8494,1115],{"class":1107},[1050,8496,1129],{"class":1091},[1050,8498,8499],{"class":1052,"line":2814},[1050,8500,8501],{"class":1091},"        },\n",[1050,8503,8504,8506,8509,8511,8513],{"class":1052,"line":2819},[1050,8505,1108],{"class":1107},[1050,8507,8508],{"class":1111},"security",[1050,8510,1115],{"class":1107},[1050,8512,1118],{"class":1091},[1050,8514,1175],{"class":1091},[1050,8516,8517,8519,8521,8523,8525,8527,8530,8533,8535,8537,8539],{"class":1052,"line":4159},[1050,8518,1181],{"class":1107},[1050,8520,8440],{"class":1111},[1050,8522,1115],{"class":1107},[1050,8524,1118],{"class":1091},[1050,8526,1121],{"class":1107},[1050,8528,8529],{"class":3477},"{asctime}",[1050,8531,8532],{"class":1111}," SECURITY ",[1050,8534,8449],{"class":3477},[1050,8536,8473],{"class":3477},[1050,8538,1115],{"class":1107},[1050,8540,1129],{"class":1091},[1050,8542,8543,8545,8547,8549,8551,8553,8555,8557],{"class":1052,"line":4168},[1050,8544,1181],{"class":1107},[1050,8546,8484],{"class":1111},[1050,8548,1115],{"class":1107},[1050,8550,1118],{"class":1091},[1050,8552,1121],{"class":1107},[1050,8554,3478],{"class":1111},[1050,8556,1115],{"class":1107},[1050,8558,1129],{"class":1091},[1050,8560,8561],{"class":1052,"line":4180},[1050,8562,8501],{"class":1091},[1050,8564,8565],{"class":1052,"line":4185},[1050,8566,1135],{"class":1091},[1050,8568,8569,8571,8574,8576,8578],{"class":1052,"line":4190},[1050,8570,1441],{"class":1107},[1050,8572,8573],{"class":1111},"handlers",[1050,8575,1115],{"class":1107},[1050,8577,1118],{"class":1091},[1050,8579,1175],{"class":1091},[1050,8581,8582,8584,8587,8589,8591],{"class":1052,"line":4195},[1050,8583,1108],{"class":1107},[1050,8585,8586],{"class":1111},"admin_security_file",[1050,8588,1115],{"class":1107},[1050,8590,1118],{"class":1091},[1050,8592,1175],{"class":1091},[1050,8594,8595,8597,8600,8602,8604,8606,8609,8611],{"class":1052,"line":4201},[1050,8596,1181],{"class":1107},[1050,8598,8599],{"class":1111},"level",[1050,8601,1115],{"class":1107},[1050,8603,1118],{"class":1091},[1050,8605,1121],{"class":1107},[1050,8607,8608],{"class":1111},"INFO",[1050,8610,1115],{"class":1107},[1050,8612,1129],{"class":1091},[1050,8614,8615,8617,8619,8621,8623,8625,8628,8630],{"class":1052,"line":4210},[1050,8616,1181],{"class":1107},[1050,8618,1566],{"class":1111},[1050,8620,1115],{"class":1107},[1050,8622,1118],{"class":1091},[1050,8624,1121],{"class":1107},[1050,8626,8627],{"class":1111},"logging.handlers.RotatingFileHandler",[1050,8629,1115],{"class":1107},[1050,8631,1129],{"class":1091},[1050,8633,8634,8636,8638,8640,8642,8644,8647,8649],{"class":1052,"line":4225},[1050,8635,1181],{"class":1107},[1050,8637,6813],{"class":1111},[1050,8639,1115],{"class":1107},[1050,8641,1118],{"class":1091},[1050,8643,1121],{"class":1107},[1050,8645,8646],{"class":1111},"/var/log/django/admin_security.log",[1050,8648,1115],{"class":1107},[1050,8650,1129],{"class":1091},[1050,8652,8653,8655,8658,8660,8662,8664,8666,8668,8670,8672,8674],{"class":1052,"line":4240},[1050,8654,1181],{"class":1107},[1050,8656,8657],{"class":1111},"maxBytes",[1050,8659,1115],{"class":1107},[1050,8661,1118],{"class":1091},[1050,8663,6206],{"class":1191},[1050,8665,6209],{"class":1658},[1050,8667,6212],{"class":1191},[1050,8669,6209],{"class":1658},[1050,8671,6212],{"class":1191},[1050,8673,1195],{"class":1091},[1050,8675,8676],{"class":1056},"  # 10MB\n",[1050,8678,8679,8681,8684,8686,8688,8691],{"class":1052,"line":4255},[1050,8680,1181],{"class":1107},[1050,8682,8683],{"class":1111},"backupCount",[1050,8685,1115],{"class":1107},[1050,8687,1118],{"class":1091},[1050,8689,8690],{"class":1191}," 5",[1050,8692,1129],{"class":1091},[1050,8694,8695,8697,8700,8702,8704,8706,8708,8710],{"class":1052,"line":7178},[1050,8696,1181],{"class":1107},[1050,8698,8699],{"class":1111},"formatter",[1050,8701,1115],{"class":1107},[1050,8703,1118],{"class":1091},[1050,8705,1121],{"class":1107},[1050,8707,8508],{"class":1111},[1050,8709,1115],{"class":1107},[1050,8711,1129],{"class":1091},[1050,8713,8714],{"class":1052,"line":7192},[1050,8715,8501],{"class":1091},[1050,8717,8718,8720,8723,8725,8727],{"class":1052,"line":7211},[1050,8719,1108],{"class":1107},[1050,8721,8722],{"class":1111},"security_email",[1050,8724,1115],{"class":1107},[1050,8726,1118],{"class":1091},[1050,8728,1175],{"class":1091},[1050,8730,8731,8733,8735,8737,8739,8741,8744,8746],{"class":1052,"line":7216},[1050,8732,1181],{"class":1107},[1050,8734,8599],{"class":1111},[1050,8736,1115],{"class":1107},[1050,8738,1118],{"class":1091},[1050,8740,1121],{"class":1107},[1050,8742,8743],{"class":1111},"WARNING",[1050,8745,1115],{"class":1107},[1050,8747,1129],{"class":1091},[1050,8749,8750,8752,8754,8756,8758,8760,8763,8765],{"class":1052,"line":7222},[1050,8751,1181],{"class":1107},[1050,8753,1566],{"class":1111},[1050,8755,1115],{"class":1107},[1050,8757,1118],{"class":1091},[1050,8759,1121],{"class":1107},[1050,8761,8762],{"class":1111},"django.utils.log.AdminEmailHandler",[1050,8764,1115],{"class":1107},[1050,8766,1129],{"class":1091},[1050,8768,8769,8771,8773,8775,8777,8779,8781,8783],{"class":1052,"line":7234},[1050,8770,1181],{"class":1107},[1050,8772,8699],{"class":1111},[1050,8774,1115],{"class":1107},[1050,8776,1118],{"class":1091},[1050,8778,1121],{"class":1107},[1050,8780,8427],{"class":1111},[1050,8782,1115],{"class":1107},[1050,8784,1129],{"class":1091},[1050,8786,8787],{"class":1052,"line":7270},[1050,8788,8501],{"class":1091},[1050,8790,8791],{"class":1052,"line":7292},[1050,8792,1135],{"class":1091},[1050,8794,8795,8797,8800,8802,8804],{"class":1052,"line":7297},[1050,8796,1441],{"class":1107},[1050,8798,8799],{"class":1111},"loggers",[1050,8801,1115],{"class":1107},[1050,8803,1118],{"class":1091},[1050,8805,1175],{"class":1091},[1050,8807,8808,8810,8812,8814,8816],{"class":1052,"line":7302},[1050,8809,1108],{"class":1107},[1050,8811,7510],{"class":1111},[1050,8813,1115],{"class":1107},[1050,8815,1118],{"class":1091},[1050,8817,1175],{"class":1091},[1050,8819,8820,8822,8824,8826,8828,8830,8832,8834,8836,8838,8840,8842,8844],{"class":1052,"line":7320},[1050,8821,1181],{"class":1107},[1050,8823,8573],{"class":1111},[1050,8825,1115],{"class":1107},[1050,8827,1118],{"class":1091},[1050,8829,2393],{"class":1091},[1050,8831,1115],{"class":1107},[1050,8833,8586],{"class":1111},[1050,8835,1115],{"class":1107},[1050,8837,1195],{"class":1091},[1050,8839,1121],{"class":1107},[1050,8841,8722],{"class":1111},[1050,8843,1115],{"class":1107},[1050,8845,7267],{"class":1091},[1050,8847,8848,8850,8852,8854,8856,8858,8860,8862],{"class":1052,"line":7347},[1050,8849,1181],{"class":1107},[1050,8851,8599],{"class":1111},[1050,8853,1115],{"class":1107},[1050,8855,1118],{"class":1091},[1050,8857,1121],{"class":1107},[1050,8859,8608],{"class":1111},[1050,8861,1115],{"class":1107},[1050,8863,1129],{"class":1091},[1050,8865,8866,8868,8871,8873,8875,8877],{"class":1052,"line":7357},[1050,8867,1181],{"class":1107},[1050,8869,8870],{"class":1111},"propagate",[1050,8872,1115],{"class":1107},[1050,8874,1118],{"class":1091},[1050,8876,8405],{"class":1297},[1050,8878,1129],{"class":1091},[1050,8880,8881],{"class":1052,"line":7389},[1050,8882,8501],{"class":1091},[1050,8884,8885],{"class":1052,"line":7416},[1050,8886,1135],{"class":1091},[1050,8888,8889],{"class":1052,"line":7421},[1050,8890,8891],{"class":1091},"}\n",[1035,8893,8895],{"id":8894},"real-time-security-monitoring","Real-time Security Monitoring",[1040,8897,8899],{"className":1042,"code":8898,"language":1044,"meta":1045,"style":1045},"# monitoring.py\nfrom django.core.management.base import BaseCommand\nfrom django.core.mail import send_mail\nfrom django.contrib.auth.models import User\nfrom django.utils import timezone\nfrom datetime import timedelta\nimport re\n\nclass Command(BaseCommand):\n    \"\"\"Monitor admin security events\"\"\"\n    \n    def handle(self, *args, **options):\n        self.check_suspicious_activity()\n        self.check_failed_logins()\n        self.check_privilege_escalations()\n        self.check_unusual_access_patterns()\n    \n    def check_suspicious_activity(self):\n        \"\"\"Check for suspicious admin activity\"\"\"\n        # Check for multiple failed login attempts\n        recent_time = timezone.now() - timedelta(hours=1)\n        \n        with open('/var/log/django/admin_security.log', 'r') as f:\n            log_content = f.read()\n        \n        # Parse failed login attempts\n        failed_attempts = re.findall(\n            r'Failed admin login attempt.*IP: ([\\d.]+).*Username: (\\w+)',\n            log_content\n        )\n        \n        # Group by IP and count attempts\n        ip_attempts = {}\n        for ip, username in failed_attempts:\n            if ip not in ip_attempts:\n                ip_attempts[ip] = []\n            ip_attempts[ip].append(username)\n        \n        # Alert on suspicious patterns\n        for ip, usernames in ip_attempts.items():\n            if len(usernames) > 10:  # More than 10 attempts\n                self.send_security_alert(\n                    f'Brute force attack detected from IP {ip}',\n                    f'Multiple failed login attempts: {len(usernames)} attempts '\n                    f'for usernames: {\", \".join(set(usernames))}'\n                )\n    \n    def check_privilege_escalations(self):\n        \"\"\"Check for unauthorized privilege changes\"\"\"\n        recent_time = timezone.now() - timedelta(hours=24)\n        \n        # Check for users granted superuser status\n        new_superusers = User.objects.filter(\n            is_superuser=True,\n            date_joined__gte=recent_time\n        )\n        \n        for user in new_superusers:\n            self.send_security_alert(\n                f'New superuser created: {user.username}',\n                f'User {user.username} was granted superuser privileges '\n                f'on {user.date_joined}'\n            )\n    \n    def send_security_alert(self, subject, message):\n        \"\"\"Send security alert email\"\"\"\n        send_mail(\n            f'SECURITY ALERT: {subject}',\n            message,\n            'security@example.com',\n            ['admin@example.com', 'security-team@example.com'],\n            fail_silently=False,\n        )\n",[1047,8900,8901,8906,8931,8951,8973,8987,8999,9006,9010,9024,9033,9037,9064,9075,9086,9097,9108,9112,9125,9134,9139,9170,9174,9207,9222,9226,9231,9248,9307,9312,9316,9320,9325,9335,9352,9368,9383,9403,9407,9412,9434,9457,9469,9486,9509,9543,9547,9551,9564,9573,9602,9606,9611,9630,9641,9651,9655,9659,9673,9683,9704,9724,9744,9748,9752,9775,9784,9791,9809,9816,9827,9850,9861],{"__ignoreMap":1045},[1050,8902,8903],{"class":1052,"line":1053},[1050,8904,8905],{"class":1056},"# monitoring.py\n",[1050,8907,8908,8910,8912,8914,8916,8918,8921,8923,8926,8928],{"class":1052,"line":1060},[1050,8909,1526],{"class":1063},[1050,8911,1749],{"class":1067},[1050,8913,1532],{"class":1091},[1050,8915,4281],{"class":1067},[1050,8917,1532],{"class":1091},[1050,8919,8920],{"class":1067},"management",[1050,8922,1532],{"class":1091},[1050,8924,8925],{"class":1067},"base ",[1050,8927,1064],{"class":1063},[1050,8929,8930],{"class":1067}," BaseCommand\n",[1050,8932,8933,8935,8937,8939,8941,8943,8946,8948],{"class":1052,"line":1071},[1050,8934,1526],{"class":1063},[1050,8936,1749],{"class":1067},[1050,8938,1532],{"class":1091},[1050,8940,4281],{"class":1067},[1050,8942,1532],{"class":1091},[1050,8944,8945],{"class":1067},"mail ",[1050,8947,1064],{"class":1063},[1050,8949,8950],{"class":1067}," send_mail\n",[1050,8952,8953,8955,8957,8959,8961,8963,8965,8967,8969,8971],{"class":1052,"line":1078},[1050,8954,1526],{"class":1063},[1050,8956,1749],{"class":1067},[1050,8958,1532],{"class":1091},[1050,8960,1754],{"class":1067},[1050,8962,1532],{"class":1091},[1050,8964,1759],{"class":1067},[1050,8966,1532],{"class":1091},[1050,8968,1764],{"class":1067},[1050,8970,1064],{"class":1063},[1050,8972,5161],{"class":1067},[1050,8974,8975,8977,8979,8981,8983,8985],{"class":1052,"line":1084},[1050,8976,1526],{"class":1063},[1050,8978,1749],{"class":1067},[1050,8980,1532],{"class":1091},[1050,8982,2271],{"class":1067},[1050,8984,1064],{"class":1063},[1050,8986,2276],{"class":1067},[1050,8988,8989,8991,8994,8996],{"class":1052,"line":1098},[1050,8990,1526],{"class":1063},[1050,8992,8993],{"class":1067}," datetime ",[1050,8995,1064],{"class":1063},[1050,8997,8998],{"class":1067}," timedelta\n",[1050,9000,9001,9003],{"class":1052,"line":1104},[1050,9002,1064],{"class":1063},[1050,9004,9005],{"class":1067}," re\n",[1050,9007,9008],{"class":1052,"line":1132},[1050,9009,1075],{"emptyLinePlaceholder":1074},[1050,9011,9012,9014,9017,9019,9022],{"class":1052,"line":1138},[1050,9013,1566],{"class":1565},[1050,9015,9016],{"class":1569}," Command",[1050,9018,1573],{"class":1091},[1050,9020,9021],{"class":1576},"BaseCommand",[1050,9023,1580],{"class":1091},[1050,9025,9026,9028,9031],{"class":1052,"line":1143},[1050,9027,1586],{"class":1585},[1050,9029,9030],{"class":1589},"Monitor admin security events",[1050,9032,1593],{"class":1585},[1050,9034,9035],{"class":1052,"line":1163},[1050,9036,1598],{"class":1067},[1050,9038,9039,9041,9044,9046,9048,9050,9052,9055,9057,9059,9062],{"class":1052,"line":1178},[1050,9040,1603],{"class":1565},[1050,9042,9043],{"class":1606}," handle",[1050,9045,1573],{"class":1091},[1050,9047,1613],{"class":1612},[1050,9049,1195],{"class":1091},[1050,9051,6209],{"class":1658},[1050,9053,9054],{"class":1618},"args",[1050,9056,1195],{"class":1091},[1050,9058,7339],{"class":1658},[1050,9060,9061],{"class":1618},"options",[1050,9063,1580],{"class":1091},[1050,9065,9066,9068,9070,9073],{"class":1052,"line":1201},[1050,9067,3351],{"class":1087},[1050,9069,1532],{"class":1091},[1050,9071,9072],{"class":1687},"check_suspicious_activity",[1050,9074,1691],{"class":1091},[1050,9076,9077,9079,9081,9084],{"class":1052,"line":1207},[1050,9078,3351],{"class":1087},[1050,9080,1532],{"class":1091},[1050,9082,9083],{"class":1687},"check_failed_logins",[1050,9085,1691],{"class":1091},[1050,9087,9088,9090,9092,9095],{"class":1052,"line":1212},[1050,9089,3351],{"class":1087},[1050,9091,1532],{"class":1091},[1050,9093,9094],{"class":1687},"check_privilege_escalations",[1050,9096,1691],{"class":1091},[1050,9098,9099,9101,9103,9106],{"class":1052,"line":1217},[1050,9100,3351],{"class":1087},[1050,9102,1532],{"class":1091},[1050,9104,9105],{"class":1687},"check_unusual_access_patterns",[1050,9107,1691],{"class":1091},[1050,9109,9110],{"class":1052,"line":1237},[1050,9111,1598],{"class":1067},[1050,9113,9114,9116,9119,9121,9123],{"class":1052,"line":1242},[1050,9115,1603],{"class":1565},[1050,9117,9118],{"class":1606}," check_suspicious_activity",[1050,9120,1573],{"class":1091},[1050,9122,1613],{"class":1612},[1050,9124,1580],{"class":1091},[1050,9126,9127,9129,9132],{"class":1052,"line":1247},[1050,9128,1626],{"class":1585},[1050,9130,9131],{"class":1589},"Check for suspicious admin activity",[1050,9133,1593],{"class":1585},[1050,9135,9136],{"class":1052,"line":1267},[1050,9137,9138],{"class":1056},"        # Check for multiple failed login attempts\n",[1050,9140,9141,9144,9146,9148,9150,9152,9154,9157,9159,9161,9164,9166,9168],{"class":1052,"line":1272},[1050,9142,9143],{"class":1067},"        recent_time ",[1050,9145,1713],{"class":1091},[1050,9147,5233],{"class":1067},[1050,9149,1532],{"class":1091},[1050,9151,5238],{"class":1687},[1050,9153,5241],{"class":1091},[1050,9155,9156],{"class":1658}," -",[1050,9158,5246],{"class":1687},[1050,9160,1573],{"class":1091},[1050,9162,9163],{"class":1720},"hours",[1050,9165,1713],{"class":1091},[1050,9167,6818],{"class":1191},[1050,9169,1733],{"class":1091},[1050,9171,9172],{"class":1052,"line":1278},[1050,9173,2641],{"class":1067},[1050,9175,9176,9179,9182,9184,9186,9188,9190,9192,9194,9197,9199,9201,9203,9205],{"class":1052,"line":1283},[1050,9177,9178],{"class":1063},"        with",[1050,9180,9181],{"class":3188}," open",[1050,9183,1573],{"class":1091},[1050,9185,1115],{"class":1107},[1050,9187,8646],{"class":1111},[1050,9189,1115],{"class":1107},[1050,9191,1195],{"class":1091},[1050,9193,1121],{"class":1107},[1050,9195,9196],{"class":1111},"r",[1050,9198,1115],{"class":1107},[1050,9200,3176],{"class":1091},[1050,9202,4095],{"class":1063},[1050,9204,4823],{"class":1067},[1050,9206,2663],{"class":1091},[1050,9208,9209,9212,9214,9216,9218,9220],{"class":1052,"line":1289},[1050,9210,9211],{"class":1067},"            log_content ",[1050,9213,1713],{"class":1091},[1050,9215,4823],{"class":1067},[1050,9217,1532],{"class":1091},[1050,9219,6492],{"class":1687},[1050,9221,1691],{"class":1091},[1050,9223,9224],{"class":1052,"line":1304},[1050,9225,2641],{"class":1067},[1050,9227,9228],{"class":1052,"line":1317},[1050,9229,9230],{"class":1056},"        # Parse failed login attempts\n",[1050,9232,9233,9236,9238,9241,9243,9246],{"class":1052,"line":1331},[1050,9234,9235],{"class":1067},"        failed_attempts ",[1050,9237,1713],{"class":1091},[1050,9239,9240],{"class":1067}," re",[1050,9242,1532],{"class":1091},[1050,9244,9245],{"class":1687},"findall",[1050,9247,2065],{"class":1091},[1050,9249,9250,9253,9255,9259,9262,9266,9269,9271,9274,9277,9280,9282,9285,9287,9289,9291,9294,9296,9299,9301,9303,9305],{"class":1052,"line":1342},[1050,9251,9252],{"class":1565},"            r",[1050,9254,1115],{"class":1107},[1050,9256,9258],{"class":9257},"s27EL","Failed admin login attempt",[1050,9260,1532],{"class":9261},"sRJPX",[1050,9263,9265],{"class":9264},"sGuSu","*",[1050,9267,9268],{"class":9257},"IP: ",[1050,9270,1573],{"class":1091},[1050,9272,7395],{"class":9273},"sETVe",[1050,9275,9276],{"class":9261},"\\d",[1050,9278,1532],{"class":9279},"sz9Cv",[1050,9281,6039],{"class":9273},[1050,9283,9284],{"class":9264},"+",[1050,9286,3176],{"class":1091},[1050,9288,1532],{"class":9261},[1050,9290,9265],{"class":9264},[1050,9292,9293],{"class":9257},"Username: ",[1050,9295,1573],{"class":1091},[1050,9297,9298],{"class":9261},"\\w",[1050,9300,9284],{"class":9264},[1050,9302,3176],{"class":1091},[1050,9304,1115],{"class":1107},[1050,9306,1129],{"class":1091},[1050,9308,9309],{"class":1052,"line":1347},[1050,9310,9311],{"class":1687},"            log_content\n",[1050,9313,9314],{"class":1052,"line":1353},[1050,9315,1696],{"class":1091},[1050,9317,9318],{"class":1052,"line":1363},[1050,9319,2641],{"class":1067},[1050,9321,9322],{"class":1052,"line":1373},[1050,9323,9324],{"class":1056},"        # Group by IP and count attempts\n",[1050,9326,9327,9330,9332],{"class":1052,"line":1383},[1050,9328,9329],{"class":1067},"        ip_attempts ",[1050,9331,1713],{"class":1091},[1050,9333,9334],{"class":1091}," {}\n",[1050,9336,9337,9339,9341,9343,9345,9347,9350],{"class":1052,"line":1388},[1050,9338,3775],{"class":1063},[1050,9340,3818],{"class":1067},[1050,9342,1195],{"class":1091},[1050,9344,4649],{"class":1067},[1050,9346,3185],{"class":1063},[1050,9348,9349],{"class":1067}," failed_attempts",[1050,9351,2663],{"class":1091},[1050,9353,9354,9356,9359,9361,9363,9366],{"class":1052,"line":1394},[1050,9355,2675],{"class":1063},[1050,9357,9358],{"class":1067}," ip ",[1050,9360,6571],{"class":1658},[1050,9362,2969],{"class":1658},[1050,9364,9365],{"class":1067}," ip_attempts",[1050,9367,2663],{"class":1091},[1050,9369,9370,9373,9375,9377,9379,9381],{"class":1052,"line":1777},[1050,9371,9372],{"class":1067},"                ip_attempts",[1050,9374,7395],{"class":1091},[1050,9376,3882],{"class":1067},[1050,9378,6039],{"class":1091},[1050,9380,1092],{"class":1091},[1050,9382,6187],{"class":1091},[1050,9384,9385,9388,9390,9392,9394,9397,9399,9401],{"class":1052,"line":1793},[1050,9386,9387],{"class":1067},"            ip_attempts",[1050,9389,7395],{"class":1091},[1050,9391,3882],{"class":1067},[1050,9393,3855],{"class":1091},[1050,9395,9396],{"class":1687},"append",[1050,9398,1573],{"class":1091},[1050,9400,2323],{"class":1687},[1050,9402,1733],{"class":1091},[1050,9404,9405],{"class":1052,"line":1809},[1050,9406,2641],{"class":1067},[1050,9408,9409],{"class":1052,"line":1814},[1050,9410,9411],{"class":1056},"        # Alert on suspicious patterns\n",[1050,9413,9414,9416,9418,9420,9423,9425,9427,9429,9432],{"class":1052,"line":1832},[1050,9415,3775],{"class":1063},[1050,9417,3818],{"class":1067},[1050,9419,1195],{"class":1091},[1050,9421,9422],{"class":1067}," usernames ",[1050,9424,3185],{"class":1063},[1050,9426,9365],{"class":1067},[1050,9428,1532],{"class":1091},[1050,9430,9431],{"class":1687},"items",[1050,9433,3112],{"class":1091},[1050,9435,9436,9438,9441,9443,9446,9448,9450,9452,9454],{"class":1052,"line":2433},[1050,9437,2675],{"class":1063},[1050,9439,9440],{"class":3188}," len",[1050,9442,1573],{"class":1091},[1050,9444,9445],{"class":1687},"usernames",[1050,9447,3176],{"class":1091},[1050,9449,6377],{"class":1658},[1050,9451,6206],{"class":1191},[1050,9453,1118],{"class":1091},[1050,9455,9456],{"class":1056},"  # More than 10 attempts\n",[1050,9458,9459,9462,9464,9467],{"class":1052,"line":2454},[1050,9460,9461],{"class":1087},"                self",[1050,9463,1532],{"class":1091},[1050,9465,9466],{"class":1687},"send_security_alert",[1050,9468,2065],{"class":1091},[1050,9470,9471,9473,9476,9478,9480,9482,9484],{"class":1052,"line":2470},[1050,9472,3554],{"class":1565},[1050,9474,9475],{"class":1111},"'Brute force attack detected from IP ",[1050,9477,3478],{"class":3477},[1050,9479,3882],{"class":1687},[1050,9481,3487],{"class":3477},[1050,9483,1115],{"class":1111},[1050,9485,1129],{"class":1091},[1050,9487,9488,9490,9493,9495,9498,9500,9502,9504,9506],{"class":1052,"line":2484},[1050,9489,3554],{"class":1565},[1050,9491,9492],{"class":1111},"'Multiple failed login attempts: ",[1050,9494,3478],{"class":3477},[1050,9496,9497],{"class":3188},"len",[1050,9499,1573],{"class":1091},[1050,9501,9445],{"class":1687},[1050,9503,3176],{"class":1091},[1050,9505,3487],{"class":3477},[1050,9507,9508],{"class":1111}," attempts '\n",[1050,9510,9511,9513,9516,9518,9520,9523,9525,9527,9529,9531,9533,9535,9537,9539,9541],{"class":1052,"line":2498},[1050,9512,3554],{"class":1565},[1050,9514,9515],{"class":1111},"'for usernames: ",[1050,9517,3478],{"class":3477},[1050,9519,3020],{"class":1107},[1050,9521,9522],{"class":1111},", ",[1050,9524,3020],{"class":1107},[1050,9526,1532],{"class":1091},[1050,9528,3158],{"class":1687},[1050,9530,1573],{"class":1091},[1050,9532,5024],{"class":2621},[1050,9534,1573],{"class":1091},[1050,9536,9445],{"class":1687},[1050,9538,3197],{"class":1091},[1050,9540,3487],{"class":3477},[1050,9542,2375],{"class":1111},[1050,9544,9545],{"class":1052,"line":2511},[1050,9546,2745],{"class":1091},[1050,9548,9549],{"class":1052,"line":2522},[1050,9550,1598],{"class":1067},[1050,9552,9553,9555,9558,9560,9562],{"class":1052,"line":2532},[1050,9554,1603],{"class":1565},[1050,9556,9557],{"class":1606}," check_privilege_escalations",[1050,9559,1573],{"class":1091},[1050,9561,1613],{"class":1612},[1050,9563,1580],{"class":1091},[1050,9565,9566,9568,9571],{"class":1052,"line":2538},[1050,9567,1626],{"class":1585},[1050,9569,9570],{"class":1589},"Check for unauthorized privilege changes",[1050,9572,1593],{"class":1585},[1050,9574,9575,9577,9579,9581,9583,9585,9587,9589,9591,9593,9595,9597,9600],{"class":1052,"line":2544},[1050,9576,9143],{"class":1067},[1050,9578,1713],{"class":1091},[1050,9580,5233],{"class":1067},[1050,9582,1532],{"class":1091},[1050,9584,5238],{"class":1687},[1050,9586,5241],{"class":1091},[1050,9588,9156],{"class":1658},[1050,9590,5246],{"class":1687},[1050,9592,1573],{"class":1091},[1050,9594,9163],{"class":1720},[1050,9596,1713],{"class":1091},[1050,9598,9599],{"class":1191},"24",[1050,9601,1733],{"class":1091},[1050,9603,9604],{"class":1052,"line":2549},[1050,9605,2641],{"class":1067},[1050,9607,9608],{"class":1052,"line":2554},[1050,9609,9610],{"class":1056},"        # Check for users granted superuser status\n",[1050,9612,9613,9616,9618,9620,9622,9624,9626,9628],{"class":1052,"line":2580},[1050,9614,9615],{"class":1067},"        new_superusers ",[1050,9617,1713],{"class":1091},[1050,9619,1769],{"class":1067},[1050,9621,1532],{"class":1091},[1050,9623,5175],{"class":1649},[1050,9625,1532],{"class":1091},[1050,9627,2712],{"class":1687},[1050,9629,2065],{"class":1091},[1050,9631,9632,9635,9637,9639],{"class":1052,"line":2585},[1050,9633,9634],{"class":1720},"            is_superuser",[1050,9636,1713],{"class":1091},[1050,9638,2100],{"class":1297},[1050,9640,1129],{"class":1091},[1050,9642,9643,9646,9648],{"class":1052,"line":2603},[1050,9644,9645],{"class":1720},"            date_joined__gte",[1050,9647,1713],{"class":1091},[1050,9649,9650],{"class":1687},"recent_time\n",[1050,9652,9653],{"class":1052,"line":2613},[1050,9654,1696],{"class":1091},[1050,9656,9657],{"class":1052,"line":2638},[1050,9658,2641],{"class":1067},[1050,9660,9661,9663,9666,9668,9671],{"class":1052,"line":2644},[1050,9662,3775],{"class":1063},[1050,9664,9665],{"class":1067}," user ",[1050,9667,3185],{"class":1063},[1050,9669,9670],{"class":1067}," new_superusers",[1050,9672,2663],{"class":1091},[1050,9674,9675,9677,9679,9681],{"class":1052,"line":2666},[1050,9676,6322],{"class":1087},[1050,9678,1532],{"class":1091},[1050,9680,9466],{"class":1687},[1050,9682,2065],{"class":1091},[1050,9684,9685,9687,9690,9692,9694,9696,9698,9700,9702],{"class":1052,"line":2672},[1050,9686,6400],{"class":1565},[1050,9688,9689],{"class":1111},"'New superuser created: ",[1050,9691,3478],{"class":3477},[1050,9693,1650],{"class":1687},[1050,9695,1532],{"class":1091},[1050,9697,2323],{"class":1649},[1050,9699,3487],{"class":3477},[1050,9701,1115],{"class":1111},[1050,9703,1129],{"class":1091},[1050,9705,9706,9708,9711,9713,9715,9717,9719,9721],{"class":1052,"line":2699},[1050,9707,6400],{"class":1565},[1050,9709,9710],{"class":1111},"'User ",[1050,9712,3478],{"class":3477},[1050,9714,1650],{"class":1687},[1050,9716,1532],{"class":1091},[1050,9718,2323],{"class":1649},[1050,9720,3487],{"class":3477},[1050,9722,9723],{"class":1111}," was granted superuser privileges '\n",[1050,9725,9726,9728,9731,9733,9735,9737,9740,9742],{"class":1052,"line":2717},[1050,9727,6400],{"class":1565},[1050,9729,9730],{"class":1111},"'on ",[1050,9732,3478],{"class":3477},[1050,9734,1650],{"class":1687},[1050,9736,1532],{"class":1091},[1050,9738,9739],{"class":1649},"date_joined",[1050,9741,3487],{"class":3477},[1050,9743,2375],{"class":1111},[1050,9745,9746],{"class":1052,"line":2742},[1050,9747,2535],{"class":1091},[1050,9749,9750],{"class":1052,"line":2748},[1050,9751,1598],{"class":1067},[1050,9753,9754,9756,9759,9761,9763,9765,9768,9770,9773],{"class":1052,"line":2774},[1050,9755,1603],{"class":1565},[1050,9757,9758],{"class":1606}," send_security_alert",[1050,9760,1573],{"class":1091},[1050,9762,1613],{"class":1612},[1050,9764,1195],{"class":1091},[1050,9766,9767],{"class":1618}," subject",[1050,9769,1195],{"class":1091},[1050,9771,9772],{"class":1618}," message",[1050,9774,1580],{"class":1091},[1050,9776,9777,9779,9782],{"class":1052,"line":2801},[1050,9778,1626],{"class":1585},[1050,9780,9781],{"class":1589},"Send security alert email",[1050,9783,1593],{"class":1585},[1050,9785,9786,9789],{"class":1052,"line":2806},[1050,9787,9788],{"class":1687},"        send_mail",[1050,9790,2065],{"class":1091},[1050,9792,9793,9795,9798,9800,9803,9805,9807],{"class":1052,"line":2814},[1050,9794,5068],{"class":1565},[1050,9796,9797],{"class":1111},"'SECURITY ALERT: ",[1050,9799,3478],{"class":3477},[1050,9801,9802],{"class":1687},"subject",[1050,9804,3487],{"class":3477},[1050,9806,1115],{"class":1111},[1050,9808,1129],{"class":1091},[1050,9810,9811,9814],{"class":1052,"line":2819},[1050,9812,9813],{"class":1687},"            message",[1050,9815,1129],{"class":1091},[1050,9817,9818,9820,9823,9825],{"class":1052,"line":4159},[1050,9819,1181],{"class":1107},[1050,9821,9822],{"class":1111},"security@example.com",[1050,9824,1115],{"class":1107},[1050,9826,1129],{"class":1091},[1050,9828,9829,9832,9834,9837,9839,9841,9843,9846,9848],{"class":1052,"line":4168},[1050,9830,9831],{"class":1091},"            [",[1050,9833,1115],{"class":1107},[1050,9835,9836],{"class":1111},"admin@example.com",[1050,9838,1115],{"class":1107},[1050,9840,1195],{"class":1091},[1050,9842,1121],{"class":1107},[1050,9844,9845],{"class":1111},"security-team@example.com",[1050,9847,1115],{"class":1107},[1050,9849,7267],{"class":1091},[1050,9851,9852,9855,9857,9859],{"class":1052,"line":4180},[1050,9853,9854],{"class":1720},"            fail_silently",[1050,9856,1713],{"class":1091},[1050,9858,4077],{"class":1297},[1050,9860,1129],{"class":1091},[1050,9862,9863],{"class":1052,"line":4185},[1050,9864,1696],{"class":1091},[1030,9866,9868],{"id":9867},"production-security-checklist","Production Security Checklist",[1035,9870,9872],{"id":9871},"environment-configuration","Environment Configuration",[1040,9874,9876],{"className":1042,"code":9875,"language":1044,"meta":1045,"style":1045},"# settings/production.py\nimport os\n\n# Security settings\nDEBUG = False\nALLOWED_HOSTS = ['yourdomain.com', 'www.yourdomain.com']\n\n# HTTPS enforcement\nSECURE_SSL_REDIRECT = True\nSECURE_PROXY_SSL_HEADER = ('HTTP_X_FORWARDED_PROTO', 'https')\nSECURE_HSTS_SECONDS = 31536000  # 1 year\nSECURE_HSTS_INCLUDE_SUBDOMAINS = True\nSECURE_HSTS_PRELOAD = True\n\n# Cookie security\nSESSION_COOKIE_SECURE = True\nCSRF_COOKIE_SECURE = True\nSECURE_BROWSER_XSS_FILTER = True\nSECURE_CONTENT_TYPE_NOSNIFF = True\n\n# Admin security\nADMIN_URL = os.environ.get('ADMIN_URL')\nif not ADMIN_URL or len(ADMIN_URL) \u003C 10:\n    raise ValueError(\"ADMIN_URL must be set and at least 10 characters long\")\n\n# Database security\nDATABASES = {\n    'default': {\n        'ENGINE': 'django.db.backends.postgresql',\n        'NAME': os.environ.get('DB_NAME'),\n        'USER': os.environ.get('DB_USER'),\n        'PASSWORD': os.environ.get('DB_PASSWORD'),\n        'HOST': os.environ.get('DB_HOST'),\n        'PORT': os.environ.get('DB_PORT', '5432'),\n        'OPTIONS': {\n            'sslmode': 'require',\n        },\n    }\n}\n\n# File upload security\nFILE_UPLOAD_MAX_MEMORY_SIZE = 5 * 1024 * 1024  # 5MB\nDATA_UPLOAD_MAX_MEMORY_SIZE = 10 * 1024 * 1024  # 10MB\nFILE_UPLOAD_PERMISSIONS = 0o644\n",[1047,9877,9878,9883,9889,9893,9898,9907,9934,9938,9943,9952,9980,9993,10002,10011,10015,10020,10028,10036,10045,10054,10058,10063,10089,10115,10132,10136,10141,10150,10162,10182,10213,10245,10277,10309,10350,10362,10382,10386,10391,10395,10399,10404,10423,10442],{"__ignoreMap":1045},[1050,9879,9880],{"class":1052,"line":1053},[1050,9881,9882],{"class":1056},"# settings/production.py\n",[1050,9884,9885,9887],{"class":1052,"line":1060},[1050,9886,1064],{"class":1063},[1050,9888,1068],{"class":1067},[1050,9890,9891],{"class":1052,"line":1071},[1050,9892,1075],{"emptyLinePlaceholder":1074},[1050,9894,9895],{"class":1052,"line":1078},[1050,9896,9897],{"class":1056},"# Security settings\n",[1050,9899,9900,9903,9905],{"class":1052,"line":1084},[1050,9901,9902],{"class":1087},"DEBUG",[1050,9904,1092],{"class":1091},[1050,9906,4148],{"class":1297},[1050,9908,9909,9912,9914,9916,9918,9921,9923,9925,9927,9930,9932],{"class":1052,"line":1098},[1050,9910,9911],{"class":1087},"ALLOWED_HOSTS",[1050,9913,1092],{"class":1091},[1050,9915,2393],{"class":1091},[1050,9917,1115],{"class":1107},[1050,9919,9920],{"class":1111},"yourdomain.com",[1050,9922,1115],{"class":1107},[1050,9924,1195],{"class":1091},[1050,9926,1121],{"class":1107},[1050,9928,9929],{"class":1111},"www.yourdomain.com",[1050,9931,1115],{"class":1107},[1050,9933,1275],{"class":1091},[1050,9935,9936],{"class":1052,"line":1104},[1050,9937,1075],{"emptyLinePlaceholder":1074},[1050,9939,9940],{"class":1052,"line":1132},[1050,9941,9942],{"class":1056},"# HTTPS enforcement\n",[1050,9944,9945,9948,9950],{"class":1052,"line":1138},[1050,9946,9947],{"class":1087},"SECURE_SSL_REDIRECT",[1050,9949,1092],{"class":1091},[1050,9951,1339],{"class":1297},[1050,9953,9954,9957,9959,9962,9964,9967,9969,9971,9973,9976,9978],{"class":1052,"line":1143},[1050,9955,9956],{"class":1087},"SECURE_PROXY_SSL_HEADER",[1050,9958,1092],{"class":1091},[1050,9960,9961],{"class":1091}," (",[1050,9963,1115],{"class":1107},[1050,9965,9966],{"class":1111},"HTTP_X_FORWARDED_PROTO",[1050,9968,1115],{"class":1107},[1050,9970,1195],{"class":1091},[1050,9972,1121],{"class":1107},[1050,9974,9975],{"class":1111},"https",[1050,9977,1115],{"class":1107},[1050,9979,1733],{"class":1091},[1050,9981,9982,9985,9987,9990],{"class":1052,"line":1163},[1050,9983,9984],{"class":1087},"SECURE_HSTS_SECONDS",[1050,9986,1092],{"class":1091},[1050,9988,9989],{"class":1191}," 31536000",[1050,9991,9992],{"class":1056},"  # 1 year\n",[1050,9994,9995,9998,10000],{"class":1052,"line":1178},[1050,9996,9997],{"class":1087},"SECURE_HSTS_INCLUDE_SUBDOMAINS",[1050,9999,1092],{"class":1091},[1050,10001,1339],{"class":1297},[1050,10003,10004,10007,10009],{"class":1052,"line":1201},[1050,10005,10006],{"class":1087},"SECURE_HSTS_PRELOAD",[1050,10008,1092],{"class":1091},[1050,10010,1339],{"class":1297},[1050,10012,10013],{"class":1052,"line":1207},[1050,10014,1075],{"emptyLinePlaceholder":1074},[1050,10016,10017],{"class":1052,"line":1212},[1050,10018,10019],{"class":1056},"# Cookie security\n",[1050,10021,10022,10024,10026],{"class":1052,"line":1217},[1050,10023,1292],{"class":1087},[1050,10025,1092],{"class":1091},[1050,10027,1339],{"class":1297},[1050,10029,10030,10032,10034],{"class":1052,"line":1237},[1050,10031,1356],{"class":1087},[1050,10033,1092],{"class":1091},[1050,10035,1339],{"class":1297},[1050,10037,10038,10041,10043],{"class":1052,"line":1242},[1050,10039,10040],{"class":1087},"SECURE_BROWSER_XSS_FILTER",[1050,10042,1092],{"class":1091},[1050,10044,1339],{"class":1297},[1050,10046,10047,10050,10052],{"class":1052,"line":1247},[1050,10048,10049],{"class":1087},"SECURE_CONTENT_TYPE_NOSNIFF",[1050,10051,1092],{"class":1091},[1050,10053,1339],{"class":1297},[1050,10055,10056],{"class":1052,"line":1267},[1050,10057,1075],{"emptyLinePlaceholder":1074},[1050,10059,10060],{"class":1052,"line":1272},[1050,10061,10062],{"class":1056},"# Admin security\n",[1050,10064,10065,10067,10069,10071,10073,10075,10077,10079,10081,10083,10085,10087],{"class":1052,"line":1278},[1050,10066,2915],{"class":1087},[1050,10068,1092],{"class":1091},[1050,10070,2920],{"class":1067},[1050,10072,1532],{"class":1091},[1050,10074,2925],{"class":1649},[1050,10076,1532],{"class":1091},[1050,10078,2930],{"class":1687},[1050,10080,1573],{"class":1091},[1050,10082,1115],{"class":1107},[1050,10084,2915],{"class":1111},[1050,10086,1115],{"class":1107},[1050,10088,1733],{"class":1091},[1050,10090,10091,10093,10095,10097,10100,10102,10104,10106,10108,10111,10113],{"class":1052,"line":1283},[1050,10092,2963],{"class":1063},[1050,10094,2650],{"class":1658},[1050,10096,2966],{"class":1087},[1050,10098,10099],{"class":1658}," or",[1050,10101,9440],{"class":3188},[1050,10103,1573],{"class":1091},[1050,10105,2915],{"class":2087},[1050,10107,3176],{"class":1091},[1050,10109,10110],{"class":1658}," \u003C",[1050,10112,6206],{"class":1191},[1050,10114,2663],{"class":1091},[1050,10116,10117,10119,10121,10123,10125,10128,10130],{"class":1052,"line":1289},[1050,10118,3012],{"class":1063},[1050,10120,3015],{"class":2621},[1050,10122,1573],{"class":1091},[1050,10124,3020],{"class":1107},[1050,10126,10127],{"class":1111},"ADMIN_URL must be set and at least 10 characters long",[1050,10129,3020],{"class":1107},[1050,10131,1733],{"class":1091},[1050,10133,10134],{"class":1052,"line":1304},[1050,10135,1075],{"emptyLinePlaceholder":1074},[1050,10137,10138],{"class":1052,"line":1317},[1050,10139,10140],{"class":1056},"# Database security\n",[1050,10142,10143,10146,10148],{"class":1052,"line":1331},[1050,10144,10145],{"class":1087},"DATABASES",[1050,10147,1092],{"class":1091},[1050,10149,1175],{"class":1091},[1050,10151,10152,10154,10156,10158,10160],{"class":1052,"line":1342},[1050,10153,1441],{"class":1107},[1050,10155,2174],{"class":1111},[1050,10157,1115],{"class":1107},[1050,10159,1118],{"class":1091},[1050,10161,1175],{"class":1091},[1050,10163,10164,10166,10169,10171,10173,10175,10178,10180],{"class":1052,"line":1347},[1050,10165,1108],{"class":1107},[1050,10167,10168],{"class":1111},"ENGINE",[1050,10170,1115],{"class":1107},[1050,10172,1118],{"class":1091},[1050,10174,1121],{"class":1107},[1050,10176,10177],{"class":1111},"django.db.backends.postgresql",[1050,10179,1115],{"class":1107},[1050,10181,1129],{"class":1091},[1050,10183,10184,10186,10188,10190,10192,10194,10196,10198,10200,10202,10204,10206,10209,10211],{"class":1052,"line":1353},[1050,10185,1108],{"class":1107},[1050,10187,1112],{"class":1111},[1050,10189,1115],{"class":1107},[1050,10191,1118],{"class":1091},[1050,10193,2920],{"class":1067},[1050,10195,1532],{"class":1091},[1050,10197,2925],{"class":1649},[1050,10199,1532],{"class":1091},[1050,10201,2930],{"class":1687},[1050,10203,1573],{"class":1091},[1050,10205,1115],{"class":1107},[1050,10207,10208],{"class":1111},"DB_NAME",[1050,10210,1115],{"class":1107},[1050,10212,1972],{"class":1091},[1050,10214,10215,10217,10220,10222,10224,10226,10228,10230,10232,10234,10236,10238,10241,10243],{"class":1052,"line":1363},[1050,10216,1108],{"class":1107},[1050,10218,10219],{"class":1111},"USER",[1050,10221,1115],{"class":1107},[1050,10223,1118],{"class":1091},[1050,10225,2920],{"class":1067},[1050,10227,1532],{"class":1091},[1050,10229,2925],{"class":1649},[1050,10231,1532],{"class":1091},[1050,10233,2930],{"class":1687},[1050,10235,1573],{"class":1091},[1050,10237,1115],{"class":1107},[1050,10239,10240],{"class":1111},"DB_USER",[1050,10242,1115],{"class":1107},[1050,10244,1972],{"class":1091},[1050,10246,10247,10249,10252,10254,10256,10258,10260,10262,10264,10266,10268,10270,10273,10275],{"class":1052,"line":1373},[1050,10248,1108],{"class":1107},[1050,10250,10251],{"class":1111},"PASSWORD",[1050,10253,1115],{"class":1107},[1050,10255,1118],{"class":1091},[1050,10257,2920],{"class":1067},[1050,10259,1532],{"class":1091},[1050,10261,2925],{"class":1649},[1050,10263,1532],{"class":1091},[1050,10265,2930],{"class":1687},[1050,10267,1573],{"class":1091},[1050,10269,1115],{"class":1107},[1050,10271,10272],{"class":1111},"DB_PASSWORD",[1050,10274,1115],{"class":1107},[1050,10276,1972],{"class":1091},[1050,10278,10279,10281,10284,10286,10288,10290,10292,10294,10296,10298,10300,10302,10305,10307],{"class":1052,"line":1383},[1050,10280,1108],{"class":1107},[1050,10282,10283],{"class":1111},"HOST",[1050,10285,1115],{"class":1107},[1050,10287,1118],{"class":1091},[1050,10289,2920],{"class":1067},[1050,10291,1532],{"class":1091},[1050,10293,2925],{"class":1649},[1050,10295,1532],{"class":1091},[1050,10297,2930],{"class":1687},[1050,10299,1573],{"class":1091},[1050,10301,1115],{"class":1107},[1050,10303,10304],{"class":1111},"DB_HOST",[1050,10306,1115],{"class":1107},[1050,10308,1972],{"class":1091},[1050,10310,10311,10313,10316,10318,10320,10322,10324,10326,10328,10330,10332,10334,10337,10339,10341,10343,10346,10348],{"class":1052,"line":1388},[1050,10312,1108],{"class":1107},[1050,10314,10315],{"class":1111},"PORT",[1050,10317,1115],{"class":1107},[1050,10319,1118],{"class":1091},[1050,10321,2920],{"class":1067},[1050,10323,1532],{"class":1091},[1050,10325,2925],{"class":1649},[1050,10327,1532],{"class":1091},[1050,10329,2930],{"class":1687},[1050,10331,1573],{"class":1091},[1050,10333,1115],{"class":1107},[1050,10335,10336],{"class":1111},"DB_PORT",[1050,10338,1115],{"class":1107},[1050,10340,1195],{"class":1091},[1050,10342,1121],{"class":1107},[1050,10344,10345],{"class":1111},"5432",[1050,10347,1115],{"class":1107},[1050,10349,1972],{"class":1091},[1050,10351,10352,10354,10356,10358,10360],{"class":1052,"line":1394},[1050,10353,1108],{"class":1107},[1050,10355,1168],{"class":1111},[1050,10357,1115],{"class":1107},[1050,10359,1118],{"class":1091},[1050,10361,1175],{"class":1091},[1050,10363,10364,10366,10369,10371,10373,10375,10378,10380],{"class":1052,"line":1777},[1050,10365,1181],{"class":1107},[1050,10367,10368],{"class":1111},"sslmode",[1050,10370,1115],{"class":1107},[1050,10372,1118],{"class":1091},[1050,10374,1121],{"class":1107},[1050,10376,10377],{"class":1111},"require",[1050,10379,1115],{"class":1107},[1050,10381,1129],{"class":1091},[1050,10383,10384],{"class":1052,"line":1793},[1050,10385,8501],{"class":1091},[1050,10387,10388],{"class":1052,"line":1809},[1050,10389,10390],{"class":1091},"    }\n",[1050,10392,10393],{"class":1052,"line":1814},[1050,10394,8891],{"class":1091},[1050,10396,10397],{"class":1052,"line":1832},[1050,10398,1075],{"emptyLinePlaceholder":1074},[1050,10400,10401],{"class":1052,"line":2433},[1050,10402,10403],{"class":1056},"# File upload security\n",[1050,10405,10406,10409,10411,10413,10415,10417,10419,10421],{"class":1052,"line":2454},[1050,10407,10408],{"class":1087},"FILE_UPLOAD_MAX_MEMORY_SIZE",[1050,10410,1092],{"class":1091},[1050,10412,8690],{"class":1191},[1050,10414,6209],{"class":1658},[1050,10416,6212],{"class":1191},[1050,10418,6209],{"class":1658},[1050,10420,6212],{"class":1191},[1050,10422,7289],{"class":1056},[1050,10424,10425,10428,10430,10432,10434,10436,10438,10440],{"class":1052,"line":2470},[1050,10426,10427],{"class":1087},"DATA_UPLOAD_MAX_MEMORY_SIZE",[1050,10429,1092],{"class":1091},[1050,10431,6206],{"class":1191},[1050,10433,6209],{"class":1658},[1050,10435,6212],{"class":1191},[1050,10437,6209],{"class":1658},[1050,10439,6212],{"class":1191},[1050,10441,8676],{"class":1056},[1050,10443,10444,10447,10449,10452],{"class":1052,"line":2484},[1050,10445,10446],{"class":1087},"FILE_UPLOAD_PERMISSIONS",[1050,10448,1092],{"class":1091},[1050,10450,10451],{"class":1565}," 0o",[1050,10453,10454],{"class":1191},"644\n",[1035,10456,10458],{"id":10457},"security-headers-middleware","Security Headers Middleware",[1040,10460,10462],{"className":1042,"code":10461,"language":1044,"meta":1045,"style":1045},"# security_middleware.py\nclass SecurityHeadersMiddleware:\n    \"\"\"Add security headers to all responses\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n    \n    def __call__(self, request):\n        response = self.get_response(request)\n        \n        # Add security headers\n        response['X-Content-Type-Options'] = 'nosniff'\n        response['X-Frame-Options'] = 'DENY'\n        response['X-XSS-Protection'] = '1; mode=block'\n        response['Referrer-Policy'] = 'strict-origin-when-cross-origin'\n        \n        # Content Security Policy for admin\n        if request.path.startswith('/admin/'):\n            response['Content-Security-Policy'] = (\n                \"default-src 'self'; \"\n                \"script-src 'self' 'unsafe-inline'; \"\n                \"style-src 'self' 'unsafe-inline'; \"\n                \"img-src 'self' data:; \"\n                \"font-src 'self'\"\n            )\n        \n        return response\n",[1047,10463,10464,10469,10478,10487,10491,10507,10519,10523,10539,10557,10561,10566,10591,10615,10639,10663,10667,10672,10697,10717,10728,10737,10746,10755,10764,10768,10772],{"__ignoreMap":1045},[1050,10465,10466],{"class":1052,"line":1053},[1050,10467,10468],{"class":1056},"# security_middleware.py\n",[1050,10470,10471,10473,10476],{"class":1052,"line":1060},[1050,10472,1566],{"class":1565},[1050,10474,10475],{"class":1569}," SecurityHeadersMiddleware",[1050,10477,2663],{"class":1091},[1050,10479,10480,10482,10485],{"class":1052,"line":1071},[1050,10481,1586],{"class":1585},[1050,10483,10484],{"class":1589},"Add security headers to all responses",[1050,10486,1593],{"class":1585},[1050,10488,10489],{"class":1052,"line":1078},[1050,10490,1598],{"class":1067},[1050,10492,10493,10495,10497,10499,10501,10503,10505],{"class":1052,"line":1084},[1050,10494,1603],{"class":1565},[1050,10496,3335],{"class":3188},[1050,10498,1573],{"class":1091},[1050,10500,1613],{"class":1612},[1050,10502,1195],{"class":1091},[1050,10504,3344],{"class":1618},[1050,10506,1580],{"class":1091},[1050,10508,10509,10511,10513,10515,10517],{"class":1052,"line":1098},[1050,10510,3351],{"class":1087},[1050,10512,1532],{"class":1091},[1050,10514,3356],{"class":1649},[1050,10516,1092],{"class":1091},[1050,10518,3361],{"class":1067},[1050,10520,10521],{"class":1052,"line":1104},[1050,10522,1598],{"class":1067},[1050,10524,10525,10527,10529,10531,10533,10535,10537],{"class":1052,"line":1132},[1050,10526,1603],{"class":1565},[1050,10528,3440],{"class":3188},[1050,10530,1573],{"class":1091},[1050,10532,1613],{"class":1612},[1050,10534,1195],{"class":1091},[1050,10536,1619],{"class":1618},[1050,10538,1580],{"class":1091},[1050,10540,10541,10543,10545,10547,10549,10551,10553,10555],{"class":1052,"line":1138},[1050,10542,7680],{"class":1067},[1050,10544,1713],{"class":1091},[1050,10546,3501],{"class":1087},[1050,10548,1532],{"class":1091},[1050,10550,3356],{"class":1687},[1050,10552,1573],{"class":1091},[1050,10554,2633],{"class":1687},[1050,10556,1733],{"class":1091},[1050,10558,10559],{"class":1052,"line":1143},[1050,10560,2641],{"class":1067},[1050,10562,10563],{"class":1052,"line":1163},[1050,10564,10565],{"class":1056},"        # Add security headers\n",[1050,10567,10568,10571,10573,10575,10578,10580,10582,10584,10586,10589],{"class":1052,"line":1178},[1050,10569,10570],{"class":1067},"        response",[1050,10572,7395],{"class":1091},[1050,10574,1115],{"class":1107},[1050,10576,10577],{"class":1111},"X-Content-Type-Options",[1050,10579,1115],{"class":1107},[1050,10581,6039],{"class":1091},[1050,10583,1092],{"class":1091},[1050,10585,1121],{"class":1107},[1050,10587,10588],{"class":1111},"nosniff",[1050,10590,2375],{"class":1107},[1050,10592,10593,10595,10597,10599,10602,10604,10606,10608,10610,10613],{"class":1052,"line":1201},[1050,10594,10570],{"class":1067},[1050,10596,7395],{"class":1091},[1050,10598,1115],{"class":1107},[1050,10600,10601],{"class":1111},"X-Frame-Options",[1050,10603,1115],{"class":1107},[1050,10605,6039],{"class":1091},[1050,10607,1092],{"class":1091},[1050,10609,1121],{"class":1107},[1050,10611,10612],{"class":1111},"DENY",[1050,10614,2375],{"class":1107},[1050,10616,10617,10619,10621,10623,10626,10628,10630,10632,10634,10637],{"class":1052,"line":1207},[1050,10618,10570],{"class":1067},[1050,10620,7395],{"class":1091},[1050,10622,1115],{"class":1107},[1050,10624,10625],{"class":1111},"X-XSS-Protection",[1050,10627,1115],{"class":1107},[1050,10629,6039],{"class":1091},[1050,10631,1092],{"class":1091},[1050,10633,1121],{"class":1107},[1050,10635,10636],{"class":1111},"1; mode=block",[1050,10638,2375],{"class":1107},[1050,10640,10641,10643,10645,10647,10650,10652,10654,10656,10658,10661],{"class":1052,"line":1212},[1050,10642,10570],{"class":1067},[1050,10644,7395],{"class":1091},[1050,10646,1115],{"class":1107},[1050,10648,10649],{"class":1111},"Referrer-Policy",[1050,10651,1115],{"class":1107},[1050,10653,6039],{"class":1091},[1050,10655,1092],{"class":1091},[1050,10657,1121],{"class":1107},[1050,10659,10660],{"class":1111},"strict-origin-when-cross-origin",[1050,10662,2375],{"class":1107},[1050,10664,10665],{"class":1052,"line":1217},[1050,10666,2641],{"class":1067},[1050,10668,10669],{"class":1052,"line":1237},[1050,10670,10671],{"class":1056},"        # Content Security Policy for admin\n",[1050,10673,10674,10676,10678,10680,10682,10684,10686,10688,10690,10693,10695],{"class":1052,"line":1242},[1050,10675,2647],{"class":1063},[1050,10677,1619],{"class":1067},[1050,10679,1532],{"class":1091},[1050,10681,3461],{"class":1649},[1050,10683,1532],{"class":1091},[1050,10685,3466],{"class":1687},[1050,10687,1573],{"class":1091},[1050,10689,1115],{"class":1107},[1050,10691,10692],{"class":1111},"/admin/",[1050,10694,1115],{"class":1107},[1050,10696,1580],{"class":1091},[1050,10698,10699,10702,10704,10706,10709,10711,10713,10715],{"class":1052,"line":1247},[1050,10700,10701],{"class":1067},"            response",[1050,10703,7395],{"class":1091},[1050,10705,1115],{"class":1107},[1050,10707,10708],{"class":1111},"Content-Security-Policy",[1050,10710,1115],{"class":1107},[1050,10712,6039],{"class":1091},[1050,10714,1092],{"class":1091},[1050,10716,1639],{"class":1091},[1050,10718,10719,10722,10725],{"class":1052,"line":1267},[1050,10720,10721],{"class":1107},"                \"",[1050,10723,10724],{"class":1111},"default-src 'self'; ",[1050,10726,10727],{"class":1107},"\"\n",[1050,10729,10730,10732,10735],{"class":1052,"line":1272},[1050,10731,10721],{"class":1107},[1050,10733,10734],{"class":1111},"script-src 'self' 'unsafe-inline'; ",[1050,10736,10727],{"class":1107},[1050,10738,10739,10741,10744],{"class":1052,"line":1278},[1050,10740,10721],{"class":1107},[1050,10742,10743],{"class":1111},"style-src 'self' 'unsafe-inline'; ",[1050,10745,10727],{"class":1107},[1050,10747,10748,10750,10753],{"class":1052,"line":1283},[1050,10749,10721],{"class":1107},[1050,10751,10752],{"class":1111},"img-src 'self' data:; ",[1050,10754,10727],{"class":1107},[1050,10756,10757,10759,10762],{"class":1052,"line":1289},[1050,10758,10721],{"class":1107},[1050,10760,10761],{"class":1111},"font-src 'self'",[1050,10763,10727],{"class":1107},[1050,10765,10766],{"class":1052,"line":1304},[1050,10767,2535],{"class":1091},[1050,10769,10770],{"class":1052,"line":1317},[1050,10771,2641],{"class":1067},[1050,10773,10774,10776],{"class":1052,"line":1331},[1050,10775,1636],{"class":1063},[1050,10777,7763],{"class":1067},[1030,10779,10781],{"id":10780},"security-testing","Security Testing",[1035,10783,10785],{"id":10784},"automated-security-tests","Automated Security Tests",[1040,10787,10789],{"className":1042,"code":10788,"language":1044,"meta":1045,"style":1045},"# security_tests.py\nfrom django.test import TestCase, Client\nfrom django.contrib.auth.models import User\nfrom django.urls import reverse\nimport time\n\nclass AdminSecurityTests(TestCase):\n    \"\"\"Test admin security measures\"\"\"\n    \n    def setUp(self):\n        self.client = Client()\n        self.admin_user = User.objects.create_superuser(\n            username='admin',\n            email='admin@example.com',\n            password='secure_password_123'\n        )\n    \n    def test_admin_url_not_predictable(self):\n        \"\"\"Test that admin URL is not easily guessable\"\"\"\n        common_urls = [\n            '/admin/',\n            '/administrator/',\n            '/manage/',\n            '/control/',\n            '/backend/'\n        ]\n        \n        for url in common_urls:\n            response = self.client.get(url)\n            # Should not return admin login page\n            self.assertNotContains(response, 'Django administration', status_code=404)\n    \n    def test_rate_limiting(self):\n        \"\"\"Test rate limiting on login attempts\"\"\"\n        login_url = reverse('admin:login')\n        \n        # Make multiple failed login attempts\n        for i in range(6):\n            response = self.client.post(login_url, {\n                'username': 'admin',\n                'password': 'wrong_password'\n            })\n        \n        # Should be rate limited after 5 attempts\n        response = self.client.post(login_url, {\n            'username': 'admin',\n            'password': 'wrong_password'\n        })\n        \n        self.assertEqual(response.status_code, 429)  # Too Many Requests\n    \n    def test_session_timeout(self):\n        \"\"\"Test admin session timeout\"\"\"\n        self.client.login(username='admin', password='secure_password_123')\n        \n        # Access admin page\n        response = self.client.get(reverse('admin:index'))\n        self.assertEqual(response.status_code, 200)\n        \n        # Simulate session timeout by manipulating session\n        session = self.client.session\n        session['_auth_user_id'] = str(self.admin_user.id)\n        session['_session_expiry'] = time.time() - 3600  # Expired 1 hour ago\n        session.save()\n        \n        # Should redirect to login\n        response = self.client.get(reverse('admin:index'))\n        self.assertRedirects(response, '/admin/login/?next=/admin/')\n    \n    def test_csrf_protection(self):\n        \"\"\"Test CSRF protection on admin forms\"\"\"\n        self.client.login(username='admin', password='secure_password_123')\n        \n        # Try to submit form without CSRF token\n        response = self.client.post(reverse('admin:auth_user_add'), {\n            'username': 'testuser',\n            'password1': 'testpass123',\n            'password2': 'testpass123'\n        })\n        \n        # Should be rejected due to missing CSRF token\n        self.assertEqual(response.status_code, 403)\n",[1047,10790,10791,10796,10817,10839,10854,10860,10864,10878,10887,10891,10904,10920,10944,10959,10974,10988,10992,10996,11009,11018,11027,11037,11048,11059,11070,11079,11083,11087,11101,11125,11130,11164,11168,11181,11190,11211,11215,11220,11238,11264,11282,11299,11304,11308,11313,11337,11355,11371,11376,11380,11407,11411,11424,11433,11472,11476,11481,11514,11537,11541,11546,11564,11600,11634,11644,11648,11653,11683,11707,11711,11724,11733,11771,11775,11780,11814,11833,11853,11870,11874,11878,11883],{"__ignoreMap":1045},[1050,10792,10793],{"class":1052,"line":1053},[1050,10794,10795],{"class":1056},"# security_tests.py\n",[1050,10797,10798,10800,10802,10804,10807,10809,10812,10814],{"class":1052,"line":1060},[1050,10799,1526],{"class":1063},[1050,10801,1749],{"class":1067},[1050,10803,1532],{"class":1091},[1050,10805,10806],{"class":1067},"test ",[1050,10808,1064],{"class":1063},[1050,10810,10811],{"class":1067}," TestCase",[1050,10813,1195],{"class":1091},[1050,10815,10816],{"class":1067}," Client\n",[1050,10818,10819,10821,10823,10825,10827,10829,10831,10833,10835,10837],{"class":1052,"line":1071},[1050,10820,1526],{"class":1063},[1050,10822,1749],{"class":1067},[1050,10824,1532],{"class":1091},[1050,10826,1754],{"class":1067},[1050,10828,1532],{"class":1091},[1050,10830,1759],{"class":1067},[1050,10832,1532],{"class":1091},[1050,10834,1764],{"class":1067},[1050,10836,1064],{"class":1063},[1050,10838,5161],{"class":1067},[1050,10840,10841,10843,10845,10847,10849,10851],{"class":1052,"line":1078},[1050,10842,1526],{"class":1063},[1050,10844,1749],{"class":1067},[1050,10846,1532],{"class":1091},[1050,10848,2891],{"class":1067},[1050,10850,1064],{"class":1063},[1050,10852,10853],{"class":1067}," reverse\n",[1050,10855,10856,10858],{"class":1052,"line":1084},[1050,10857,1064],{"class":1063},[1050,10859,4333],{"class":1067},[1050,10861,10862],{"class":1052,"line":1098},[1050,10863,1075],{"emptyLinePlaceholder":1074},[1050,10865,10866,10868,10871,10873,10876],{"class":1052,"line":1104},[1050,10867,1566],{"class":1565},[1050,10869,10870],{"class":1569}," AdminSecurityTests",[1050,10872,1573],{"class":1091},[1050,10874,10875],{"class":1576},"TestCase",[1050,10877,1580],{"class":1091},[1050,10879,10880,10882,10885],{"class":1052,"line":1132},[1050,10881,1586],{"class":1585},[1050,10883,10884],{"class":1589},"Test admin security measures",[1050,10886,1593],{"class":1585},[1050,10888,10889],{"class":1052,"line":1138},[1050,10890,1598],{"class":1067},[1050,10892,10893,10895,10898,10900,10902],{"class":1052,"line":1143},[1050,10894,1603],{"class":1565},[1050,10896,10897],{"class":1606}," setUp",[1050,10899,1573],{"class":1091},[1050,10901,1613],{"class":1612},[1050,10903,1580],{"class":1091},[1050,10905,10906,10908,10910,10913,10915,10918],{"class":1052,"line":1163},[1050,10907,3351],{"class":1087},[1050,10909,1532],{"class":1091},[1050,10911,10912],{"class":1649},"client",[1050,10914,1092],{"class":1091},[1050,10916,10917],{"class":1687}," Client",[1050,10919,1691],{"class":1091},[1050,10921,10922,10924,10926,10929,10931,10933,10935,10937,10939,10942],{"class":1052,"line":1178},[1050,10923,3351],{"class":1087},[1050,10925,1532],{"class":1091},[1050,10927,10928],{"class":1649},"admin_user",[1050,10930,1092],{"class":1091},[1050,10932,1769],{"class":1067},[1050,10934,1532],{"class":1091},[1050,10936,5175],{"class":1649},[1050,10938,1532],{"class":1091},[1050,10940,10941],{"class":1687},"create_superuser",[1050,10943,2065],{"class":1091},[1050,10945,10946,10949,10951,10953,10955,10957],{"class":1052,"line":1201},[1050,10947,10948],{"class":1720},"            username",[1050,10950,1713],{"class":1091},[1050,10952,1115],{"class":1107},[1050,10954,2822],{"class":1111},[1050,10956,1115],{"class":1107},[1050,10958,1129],{"class":1091},[1050,10960,10961,10964,10966,10968,10970,10972],{"class":1052,"line":1207},[1050,10962,10963],{"class":1720},"            email",[1050,10965,1713],{"class":1091},[1050,10967,1115],{"class":1107},[1050,10969,9836],{"class":1111},[1050,10971,1115],{"class":1107},[1050,10973,1129],{"class":1091},[1050,10975,10976,10979,10981,10983,10986],{"class":1052,"line":1212},[1050,10977,10978],{"class":1720},"            password",[1050,10980,1713],{"class":1091},[1050,10982,1115],{"class":1107},[1050,10984,10985],{"class":1111},"secure_password_123",[1050,10987,2375],{"class":1107},[1050,10989,10990],{"class":1052,"line":1217},[1050,10991,1696],{"class":1091},[1050,10993,10994],{"class":1052,"line":1237},[1050,10995,1598],{"class":1067},[1050,10997,10998,11000,11003,11005,11007],{"class":1052,"line":1242},[1050,10999,1603],{"class":1565},[1050,11001,11002],{"class":1606}," test_admin_url_not_predictable",[1050,11004,1573],{"class":1091},[1050,11006,1613],{"class":1612},[1050,11008,1580],{"class":1091},[1050,11010,11011,11013,11016],{"class":1052,"line":1247},[1050,11012,1626],{"class":1585},[1050,11014,11015],{"class":1589},"Test that admin URL is not easily guessable",[1050,11017,1593],{"class":1585},[1050,11019,11020,11023,11025],{"class":1052,"line":1267},[1050,11021,11022],{"class":1067},"        common_urls ",[1050,11024,1713],{"class":1091},[1050,11026,1095],{"class":1091},[1050,11028,11029,11031,11033,11035],{"class":1052,"line":1272},[1050,11030,1181],{"class":1107},[1050,11032,10692],{"class":1111},[1050,11034,1115],{"class":1107},[1050,11036,1129],{"class":1091},[1050,11038,11039,11041,11044,11046],{"class":1052,"line":1278},[1050,11040,1181],{"class":1107},[1050,11042,11043],{"class":1111},"/administrator/",[1050,11045,1115],{"class":1107},[1050,11047,1129],{"class":1091},[1050,11049,11050,11052,11055,11057],{"class":1052,"line":1283},[1050,11051,1181],{"class":1107},[1050,11053,11054],{"class":1111},"/manage/",[1050,11056,1115],{"class":1107},[1050,11058,1129],{"class":1091},[1050,11060,11061,11063,11066,11068],{"class":1052,"line":1289},[1050,11062,1181],{"class":1107},[1050,11064,11065],{"class":1111},"/control/",[1050,11067,1115],{"class":1107},[1050,11069,1129],{"class":1091},[1050,11071,11072,11074,11077],{"class":1052,"line":1304},[1050,11073,1181],{"class":1107},[1050,11075,11076],{"class":1111},"/backend/",[1050,11078,2375],{"class":1107},[1050,11080,11081],{"class":1052,"line":1317},[1050,11082,3766],{"class":1091},[1050,11084,11085],{"class":1052,"line":1331},[1050,11086,2641],{"class":1067},[1050,11088,11089,11091,11094,11096,11099],{"class":1052,"line":1342},[1050,11090,3775],{"class":1063},[1050,11092,11093],{"class":1067}," url ",[1050,11095,3185],{"class":1063},[1050,11097,11098],{"class":1067}," common_urls",[1050,11100,2663],{"class":1091},[1050,11102,11103,11106,11108,11110,11112,11114,11116,11118,11120,11123],{"class":1052,"line":1347},[1050,11104,11105],{"class":1067},"            response ",[1050,11107,1713],{"class":1091},[1050,11109,3501],{"class":1087},[1050,11111,1532],{"class":1091},[1050,11113,10912],{"class":1649},[1050,11115,1532],{"class":1091},[1050,11117,2930],{"class":1687},[1050,11119,1573],{"class":1091},[1050,11121,11122],{"class":1687},"url",[1050,11124,1733],{"class":1091},[1050,11126,11127],{"class":1052,"line":1353},[1050,11128,11129],{"class":1056},"            # Should not return admin login page\n",[1050,11131,11132,11134,11136,11139,11141,11143,11145,11147,11150,11152,11154,11157,11159,11162],{"class":1052,"line":1363},[1050,11133,6322],{"class":1087},[1050,11135,1532],{"class":1091},[1050,11137,11138],{"class":1687},"assertNotContains",[1050,11140,1573],{"class":1091},[1050,11142,8264],{"class":1687},[1050,11144,1195],{"class":1091},[1050,11146,1121],{"class":1107},[1050,11148,11149],{"class":1111},"Django administration",[1050,11151,1115],{"class":1107},[1050,11153,1195],{"class":1091},[1050,11155,11156],{"class":1720}," status_code",[1050,11158,1713],{"class":1091},[1050,11160,11161],{"class":1191},"404",[1050,11163,1733],{"class":1091},[1050,11165,11166],{"class":1052,"line":1373},[1050,11167,1598],{"class":1067},[1050,11169,11170,11172,11175,11177,11179],{"class":1052,"line":1383},[1050,11171,1603],{"class":1565},[1050,11173,11174],{"class":1606}," test_rate_limiting",[1050,11176,1573],{"class":1091},[1050,11178,1613],{"class":1612},[1050,11180,1580],{"class":1091},[1050,11182,11183,11185,11188],{"class":1052,"line":1388},[1050,11184,1626],{"class":1585},[1050,11186,11187],{"class":1589},"Test rate limiting on login attempts",[1050,11189,1593],{"class":1585},[1050,11191,11192,11195,11197,11200,11202,11204,11207,11209],{"class":1052,"line":1394},[1050,11193,11194],{"class":1067},"        login_url ",[1050,11196,1713],{"class":1091},[1050,11198,11199],{"class":1687}," reverse",[1050,11201,1573],{"class":1091},[1050,11203,1115],{"class":1107},[1050,11205,11206],{"class":1111},"admin:login",[1050,11208,1115],{"class":1107},[1050,11210,1733],{"class":1091},[1050,11212,11213],{"class":1052,"line":1777},[1050,11214,2641],{"class":1067},[1050,11216,11217],{"class":1052,"line":1793},[1050,11218,11219],{"class":1056},"        # Make multiple failed login attempts\n",[1050,11221,11222,11224,11227,11229,11231,11233,11236],{"class":1052,"line":1809},[1050,11223,3775],{"class":1063},[1050,11225,11226],{"class":1067}," i ",[1050,11228,3185],{"class":1063},[1050,11230,3189],{"class":3188},[1050,11232,1573],{"class":1091},[1050,11234,11235],{"class":1191},"6",[1050,11237,1580],{"class":1091},[1050,11239,11240,11242,11244,11246,11248,11250,11252,11255,11257,11260,11262],{"class":1052,"line":1814},[1050,11241,11105],{"class":1067},[1050,11243,1713],{"class":1091},[1050,11245,3501],{"class":1087},[1050,11247,1532],{"class":1091},[1050,11249,10912],{"class":1649},[1050,11251,1532],{"class":1091},[1050,11253,11254],{"class":1687},"post",[1050,11256,1573],{"class":1091},[1050,11258,11259],{"class":1687},"login_url",[1050,11261,1195],{"class":1091},[1050,11263,1175],{"class":1091},[1050,11265,11266,11268,11270,11272,11274,11276,11278,11280],{"class":1052,"line":1832},[1050,11267,2487],{"class":1107},[1050,11269,2323],{"class":1111},[1050,11271,1115],{"class":1107},[1050,11273,1118],{"class":1091},[1050,11275,1121],{"class":1107},[1050,11277,2822],{"class":1111},[1050,11279,1115],{"class":1107},[1050,11281,1129],{"class":1091},[1050,11283,11284,11286,11288,11290,11292,11294,11297],{"class":1052,"line":2433},[1050,11285,2487],{"class":1107},[1050,11287,4633],{"class":1111},[1050,11289,1115],{"class":1107},[1050,11291,1118],{"class":1091},[1050,11293,1121],{"class":1107},[1050,11295,11296],{"class":1111},"wrong_password",[1050,11298,2375],{"class":1107},[1050,11300,11301],{"class":1052,"line":2454},[1050,11302,11303],{"class":1091},"            })\n",[1050,11305,11306],{"class":1052,"line":2470},[1050,11307,2641],{"class":1067},[1050,11309,11310],{"class":1052,"line":2484},[1050,11311,11312],{"class":1056},"        # Should be rate limited after 5 attempts\n",[1050,11314,11315,11317,11319,11321,11323,11325,11327,11329,11331,11333,11335],{"class":1052,"line":2498},[1050,11316,7680],{"class":1067},[1050,11318,1713],{"class":1091},[1050,11320,3501],{"class":1087},[1050,11322,1532],{"class":1091},[1050,11324,10912],{"class":1649},[1050,11326,1532],{"class":1091},[1050,11328,11254],{"class":1687},[1050,11330,1573],{"class":1091},[1050,11332,11259],{"class":1687},[1050,11334,1195],{"class":1091},[1050,11336,1175],{"class":1091},[1050,11338,11339,11341,11343,11345,11347,11349,11351,11353],{"class":1052,"line":2511},[1050,11340,1181],{"class":1107},[1050,11342,2323],{"class":1111},[1050,11344,1115],{"class":1107},[1050,11346,1118],{"class":1091},[1050,11348,1121],{"class":1107},[1050,11350,2822],{"class":1111},[1050,11352,1115],{"class":1107},[1050,11354,1129],{"class":1091},[1050,11356,11357,11359,11361,11363,11365,11367,11369],{"class":1052,"line":2522},[1050,11358,1181],{"class":1107},[1050,11360,4633],{"class":1111},[1050,11362,1115],{"class":1107},[1050,11364,1118],{"class":1091},[1050,11366,1121],{"class":1107},[1050,11368,11296],{"class":1111},[1050,11370,2375],{"class":1107},[1050,11372,11373],{"class":1052,"line":2532},[1050,11374,11375],{"class":1091},"        })\n",[1050,11377,11378],{"class":1052,"line":2538},[1050,11379,2641],{"class":1067},[1050,11381,11382,11384,11386,11389,11391,11393,11395,11397,11399,11402,11404],{"class":1052,"line":2544},[1050,11383,3351],{"class":1087},[1050,11385,1532],{"class":1091},[1050,11387,11388],{"class":1687},"assertEqual",[1050,11390,1573],{"class":1091},[1050,11392,8264],{"class":1687},[1050,11394,1532],{"class":1091},[1050,11396,8234],{"class":1649},[1050,11398,1195],{"class":1091},[1050,11400,11401],{"class":1191}," 429",[1050,11403,3176],{"class":1091},[1050,11405,11406],{"class":1056},"  # Too Many Requests\n",[1050,11408,11409],{"class":1052,"line":2549},[1050,11410,1598],{"class":1067},[1050,11412,11413,11415,11418,11420,11422],{"class":1052,"line":2554},[1050,11414,1603],{"class":1565},[1050,11416,11417],{"class":1606}," test_session_timeout",[1050,11419,1573],{"class":1091},[1050,11421,1613],{"class":1612},[1050,11423,1580],{"class":1091},[1050,11425,11426,11428,11431],{"class":1052,"line":2580},[1050,11427,1626],{"class":1585},[1050,11429,11430],{"class":1589},"Test admin session timeout",[1050,11432,1593],{"class":1585},[1050,11434,11435,11437,11439,11441,11443,11446,11448,11450,11452,11454,11456,11458,11460,11462,11464,11466,11468,11470],{"class":1052,"line":2585},[1050,11436,3351],{"class":1087},[1050,11438,1532],{"class":1091},[1050,11440,10912],{"class":1649},[1050,11442,1532],{"class":1091},[1050,11444,11445],{"class":1687},"login",[1050,11447,1573],{"class":1091},[1050,11449,2323],{"class":1720},[1050,11451,1713],{"class":1091},[1050,11453,1115],{"class":1107},[1050,11455,2822],{"class":1111},[1050,11457,1115],{"class":1107},[1050,11459,1195],{"class":1091},[1050,11461,4655],{"class":1720},[1050,11463,1713],{"class":1091},[1050,11465,1115],{"class":1107},[1050,11467,10985],{"class":1111},[1050,11469,1115],{"class":1107},[1050,11471,1733],{"class":1091},[1050,11473,11474],{"class":1052,"line":2603},[1050,11475,2641],{"class":1067},[1050,11477,11478],{"class":1052,"line":2613},[1050,11479,11480],{"class":1056},"        # Access admin page\n",[1050,11482,11483,11485,11487,11489,11491,11493,11495,11497,11499,11502,11504,11506,11509,11511],{"class":1052,"line":2638},[1050,11484,7680],{"class":1067},[1050,11486,1713],{"class":1091},[1050,11488,3501],{"class":1087},[1050,11490,1532],{"class":1091},[1050,11492,10912],{"class":1649},[1050,11494,1532],{"class":1091},[1050,11496,2930],{"class":1687},[1050,11498,1573],{"class":1091},[1050,11500,11501],{"class":1687},"reverse",[1050,11503,1573],{"class":1091},[1050,11505,1115],{"class":1107},[1050,11507,11508],{"class":1111},"admin:index",[1050,11510,1115],{"class":1107},[1050,11512,11513],{"class":1091},"))\n",[1050,11515,11516,11518,11520,11522,11524,11526,11528,11530,11532,11535],{"class":1052,"line":2644},[1050,11517,3351],{"class":1087},[1050,11519,1532],{"class":1091},[1050,11521,11388],{"class":1687},[1050,11523,1573],{"class":1091},[1050,11525,8264],{"class":1687},[1050,11527,1532],{"class":1091},[1050,11529,8234],{"class":1649},[1050,11531,1195],{"class":1091},[1050,11533,11534],{"class":1191}," 200",[1050,11536,1733],{"class":1091},[1050,11538,11539],{"class":1052,"line":2666},[1050,11540,2641],{"class":1067},[1050,11542,11543],{"class":1052,"line":2672},[1050,11544,11545],{"class":1056},"        # Simulate session timeout by manipulating session\n",[1050,11547,11548,11551,11553,11555,11557,11559,11561],{"class":1052,"line":2699},[1050,11549,11550],{"class":1067},"        session ",[1050,11552,1713],{"class":1091},[1050,11554,3501],{"class":1087},[1050,11556,1532],{"class":1091},[1050,11558,10912],{"class":1649},[1050,11560,1532],{"class":1091},[1050,11562,11563],{"class":1649},"session\n",[1050,11565,11566,11569,11571,11573,11576,11578,11580,11582,11585,11587,11589,11591,11593,11595,11598],{"class":1052,"line":2717},[1050,11567,11568],{"class":1067},"        session",[1050,11570,7395],{"class":1091},[1050,11572,1115],{"class":1107},[1050,11574,11575],{"class":1111},"_auth_user_id",[1050,11577,1115],{"class":1107},[1050,11579,6039],{"class":1091},[1050,11581,1092],{"class":1091},[1050,11583,11584],{"class":2621}," str",[1050,11586,1573],{"class":1091},[1050,11588,1613],{"class":1087},[1050,11590,1532],{"class":1091},[1050,11592,10928],{"class":1649},[1050,11594,1532],{"class":1091},[1050,11596,11597],{"class":1649},"id",[1050,11599,1733],{"class":1091},[1050,11601,11602,11604,11606,11608,11611,11613,11615,11617,11620,11622,11625,11627,11629,11631],{"class":1052,"line":2742},[1050,11603,11568],{"class":1067},[1050,11605,7395],{"class":1091},[1050,11607,1115],{"class":1107},[1050,11609,11610],{"class":1111},"_session_expiry",[1050,11612,1115],{"class":1107},[1050,11614,6039],{"class":1091},[1050,11616,1092],{"class":1091},[1050,11618,11619],{"class":1067}," time",[1050,11621,1532],{"class":1091},[1050,11623,11624],{"class":1687},"time",[1050,11626,5241],{"class":1091},[1050,11628,9156],{"class":1658},[1050,11630,1325],{"class":1191},[1050,11632,11633],{"class":1056},"  # Expired 1 hour ago\n",[1050,11635,11636,11638,11640,11642],{"class":1052,"line":2748},[1050,11637,11568],{"class":1067},[1050,11639,1532],{"class":1091},[1050,11641,5276],{"class":1687},[1050,11643,1691],{"class":1091},[1050,11645,11646],{"class":1052,"line":2774},[1050,11647,2641],{"class":1067},[1050,11649,11650],{"class":1052,"line":2801},[1050,11651,11652],{"class":1056},"        # Should redirect to login\n",[1050,11654,11655,11657,11659,11661,11663,11665,11667,11669,11671,11673,11675,11677,11679,11681],{"class":1052,"line":2806},[1050,11656,7680],{"class":1067},[1050,11658,1713],{"class":1091},[1050,11660,3501],{"class":1087},[1050,11662,1532],{"class":1091},[1050,11664,10912],{"class":1649},[1050,11666,1532],{"class":1091},[1050,11668,2930],{"class":1687},[1050,11670,1573],{"class":1091},[1050,11672,11501],{"class":1687},[1050,11674,1573],{"class":1091},[1050,11676,1115],{"class":1107},[1050,11678,11508],{"class":1111},[1050,11680,1115],{"class":1107},[1050,11682,11513],{"class":1091},[1050,11684,11685,11687,11689,11692,11694,11696,11698,11700,11703,11705],{"class":1052,"line":2814},[1050,11686,3351],{"class":1087},[1050,11688,1532],{"class":1091},[1050,11690,11691],{"class":1687},"assertRedirects",[1050,11693,1573],{"class":1091},[1050,11695,8264],{"class":1687},[1050,11697,1195],{"class":1091},[1050,11699,1121],{"class":1107},[1050,11701,11702],{"class":1111},"/admin/login/?next=/admin/",[1050,11704,1115],{"class":1107},[1050,11706,1733],{"class":1091},[1050,11708,11709],{"class":1052,"line":2819},[1050,11710,1598],{"class":1067},[1050,11712,11713,11715,11718,11720,11722],{"class":1052,"line":4159},[1050,11714,1603],{"class":1565},[1050,11716,11717],{"class":1606}," test_csrf_protection",[1050,11719,1573],{"class":1091},[1050,11721,1613],{"class":1612},[1050,11723,1580],{"class":1091},[1050,11725,11726,11728,11731],{"class":1052,"line":4168},[1050,11727,1626],{"class":1585},[1050,11729,11730],{"class":1589},"Test CSRF protection on admin forms",[1050,11732,1593],{"class":1585},[1050,11734,11735,11737,11739,11741,11743,11745,11747,11749,11751,11753,11755,11757,11759,11761,11763,11765,11767,11769],{"class":1052,"line":4180},[1050,11736,3351],{"class":1087},[1050,11738,1532],{"class":1091},[1050,11740,10912],{"class":1649},[1050,11742,1532],{"class":1091},[1050,11744,11445],{"class":1687},[1050,11746,1573],{"class":1091},[1050,11748,2323],{"class":1720},[1050,11750,1713],{"class":1091},[1050,11752,1115],{"class":1107},[1050,11754,2822],{"class":1111},[1050,11756,1115],{"class":1107},[1050,11758,1195],{"class":1091},[1050,11760,4655],{"class":1720},[1050,11762,1713],{"class":1091},[1050,11764,1115],{"class":1107},[1050,11766,10985],{"class":1111},[1050,11768,1115],{"class":1107},[1050,11770,1733],{"class":1091},[1050,11772,11773],{"class":1052,"line":4185},[1050,11774,2641],{"class":1067},[1050,11776,11777],{"class":1052,"line":4190},[1050,11778,11779],{"class":1056},"        # Try to submit form without CSRF token\n",[1050,11781,11782,11784,11786,11788,11790,11792,11794,11796,11798,11800,11802,11804,11807,11809,11812],{"class":1052,"line":4195},[1050,11783,7680],{"class":1067},[1050,11785,1713],{"class":1091},[1050,11787,3501],{"class":1087},[1050,11789,1532],{"class":1091},[1050,11791,10912],{"class":1649},[1050,11793,1532],{"class":1091},[1050,11795,11254],{"class":1687},[1050,11797,1573],{"class":1091},[1050,11799,11501],{"class":1687},[1050,11801,1573],{"class":1091},[1050,11803,1115],{"class":1107},[1050,11805,11806],{"class":1111},"admin:auth_user_add",[1050,11808,1115],{"class":1107},[1050,11810,11811],{"class":1091},"),",[1050,11813,1175],{"class":1091},[1050,11815,11816,11818,11820,11822,11824,11826,11829,11831],{"class":1052,"line":4201},[1050,11817,1181],{"class":1107},[1050,11819,2323],{"class":1111},[1050,11821,1115],{"class":1107},[1050,11823,1118],{"class":1091},[1050,11825,1121],{"class":1107},[1050,11827,11828],{"class":1111},"testuser",[1050,11830,1115],{"class":1107},[1050,11832,1129],{"class":1091},[1050,11834,11835,11837,11840,11842,11844,11846,11849,11851],{"class":1052,"line":4210},[1050,11836,1181],{"class":1107},[1050,11838,11839],{"class":1111},"password1",[1050,11841,1115],{"class":1107},[1050,11843,1118],{"class":1091},[1050,11845,1121],{"class":1107},[1050,11847,11848],{"class":1111},"testpass123",[1050,11850,1115],{"class":1107},[1050,11852,1129],{"class":1091},[1050,11854,11855,11857,11860,11862,11864,11866,11868],{"class":1052,"line":4225},[1050,11856,1181],{"class":1107},[1050,11858,11859],{"class":1111},"password2",[1050,11861,1115],{"class":1107},[1050,11863,1118],{"class":1091},[1050,11865,1121],{"class":1107},[1050,11867,11848],{"class":1111},[1050,11869,2375],{"class":1107},[1050,11871,11872],{"class":1052,"line":4240},[1050,11873,11375],{"class":1091},[1050,11875,11876],{"class":1052,"line":4255},[1050,11877,2641],{"class":1067},[1050,11879,11880],{"class":1052,"line":7178},[1050,11881,11882],{"class":1056},"        # Should be rejected due to missing CSRF token\n",[1050,11884,11885,11887,11889,11891,11893,11895,11897,11899,11901,11904],{"class":1052,"line":7192},[1050,11886,3351],{"class":1087},[1050,11888,1532],{"class":1091},[1050,11890,11388],{"class":1687},[1050,11892,1573],{"class":1091},[1050,11894,8264],{"class":1687},[1050,11896,1532],{"class":1091},[1050,11898,8234],{"class":1649},[1050,11900,1195],{"class":1091},[1050,11902,11903],{"class":1191}," 403",[1050,11905,1733],{"class":1091},[1030,11907,11909],{"id":11908},"best-practices-summary","Best Practices Summary",[1035,11911,11913],{"id":11912},"authentication","Authentication",[5604,11915,11916,11919,11922,11925],{},[5613,11917,11918],{},"Implement strong password policies",[5613,11920,11921],{},"Enable two-factor authentication",[5613,11923,11924],{},"Use role-based access control",[5613,11926,11927],{},"Implement account lockout mechanisms",[1035,11929,11931],{"id":11930},"network-security","Network Security",[5604,11933,11934,11937,11940,11943],{},[5613,11935,11936],{},"Use non-predictable admin URLs",[5613,11938,11939],{},"Implement IP address restrictions",[5613,11941,11942],{},"Enable HTTPS with proper headers",[5613,11944,11945],{},"Use rate limiting and DDoS protection",[1035,11947,11949],{"id":11948},"data-protection","Data Protection",[5604,11951,11952,11955,11958,11961],{},[5613,11953,11954],{},"Sanitize all user inputs",[5613,11956,11957],{},"Validate file uploads thoroughly",[5613,11959,11960],{},"Implement proper permission checks",[5613,11962,11963],{},"Use secure session management",[1035,11965,11967],{"id":11966},"monitoring","Monitoring",[5604,11969,11970,11973,11976,11979],{},[5613,11971,11972],{},"Log all admin activities",[5613,11974,11975],{},"Monitor for suspicious patterns",[5613,11977,11978],{},"Set up real-time alerts",[5613,11980,11981],{},"Regular security audits",[1035,11983,11985],{"id":11984},"infrastructure","Infrastructure",[5604,11987,11988,11991,11994,11997],{},[5613,11989,11990],{},"Keep Django and dependencies updated",[5613,11992,11993],{},"Use secure server configurations",[5613,11995,11996],{},"Implement proper backup strategies",[5613,11998,11999],{},"Regular security testing",[1026,12001,12002],{},"By implementing these security measures, you can significantly reduce the risk of unauthorized access and protect your Django admin interface from common attack vectors. Remember that security is an ongoing process that requires regular updates and monitoring.",[8484,12004,12005],{},"html pre.shiki code .s9Tkl, html code.shiki .s9Tkl{--shiki-light:#90A4AE;--shiki-light-font-style:italic;--shiki-default:#A0ADA0;--shiki-default-font-style:inherit;--shiki-dark:#758575DD;--shiki-dark-font-style:inherit}html pre.shiki code .siDh9, html code.shiki .siDh9{--shiki-light:#39ADB5;--shiki-light-font-style:italic;--shiki-default:#1E754F;--shiki-default-font-style:inherit;--shiki-dark:#4D9375;--shiki-dark-font-style:inherit}html pre.shiki code .sftqT, html code.shiki .sftqT{--shiki-light:#90A4AE;--shiki-default:#393A34;--shiki-dark:#DBD7CAEE}html pre.shiki code .se3Ec, html code.shiki .se3Ec{--shiki-light:#90A4AE;--shiki-default:#A65E2B;--shiki-dark:#C99076}html pre.shiki code .soVBu, html code.shiki .soVBu{--shiki-light:#39ADB5;--shiki-default:#999999;--shiki-dark:#666666}html pre.shiki code .sbYkP, html code.shiki .sbYkP{--shiki-light:#39ADB5;--shiki-default:#B5695977;--shiki-dark:#C98A7D77}html pre.shiki code .sTbE_, html code.shiki .sTbE_{--shiki-light:#91B859;--shiki-default:#B56959;--shiki-dark:#C98A7D}html pre.shiki code .s7CZa, html code.shiki .s7CZa{--shiki-light:#F76D47;--shiki-default:#2F798A;--shiki-dark:#4C9A91}html pre.shiki code .s8XtY, html code.shiki .s8XtY{--shiki-light:#39ADB5;--shiki-default:#1E754F;--shiki-dark:#4D9375}html .light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html.light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .s5Kfy, html code.shiki .s5Kfy{--shiki-light:#9C3EDA;--shiki-default:#AB5959;--shiki-dark:#CB7676}html pre.shiki code .sD-vU, html code.shiki .sD-vU{--shiki-light:#E2931D;--shiki-default:#2E8F82;--shiki-dark:#5DA994}html pre.shiki code .sYn-s, html code.shiki .sYn-s{--shiki-light:#E2931D;--shiki-default:#59873A;--shiki-dark:#80A665}html pre.shiki code .sm7ve, html code.shiki .sm7ve{--shiki-light:#39ADB5;--shiki-light-font-style:italic;--shiki-default:#B5695977;--shiki-default-font-style:inherit;--shiki-dark:#C98A7D77;--shiki-dark-font-style:inherit}html pre.shiki code .sVyVU, html code.shiki .sVyVU{--shiki-light:#90A4AE;--shiki-light-font-style:italic;--shiki-default:#B56959;--shiki-default-font-style:inherit;--shiki-dark:#C98A7D;--shiki-dark-font-style:inherit}html pre.shiki code .sljsM, html code.shiki .sljsM{--shiki-light:#6182B8;--shiki-default:#59873A;--shiki-dark:#80A665}html pre.shiki code .sRjD_, html code.shiki .sRjD_{--shiki-light:#E53935;--shiki-light-font-style:italic;--shiki-default:#393A34;--shiki-default-font-style:inherit;--shiki-dark:#DBD7CAEE;--shiki-dark-font-style:inherit}html pre.shiki code .sCyAa, html code.shiki .sCyAa{--shiki-light:#90A4AE;--shiki-light-font-style:italic;--shiki-default:#393A34;--shiki-default-font-style:inherit;--shiki-dark:#DBD7CAEE;--shiki-dark-font-style:inherit}html pre.shiki code .sBPpx, html code.shiki .sBPpx{--shiki-light:#E53935;--shiki-default:#393A34;--shiki-dark:#DBD7CAEE}html pre.shiki code .sVsLi, html code.shiki .sVsLi{--shiki-light:#39ADB5;--shiki-default:#AB5959;--shiki-dark:#CB7676}html pre.shiki code .siWMO, html code.shiki .siWMO{--shiki-light:#6182B8;--shiki-default:#393A34;--shiki-dark:#DBD7CAEE}html pre.shiki code .sqOPj, html code.shiki .sqOPj{--shiki-light:#90A4AE;--shiki-light-font-style:italic;--shiki-default:#B07D48;--shiki-default-font-style:inherit;--shiki-dark:#BD976A;--shiki-dark-font-style:inherit}html pre.shiki code .s6DHA, html code.shiki .s6DHA{--shiki-light:#6182B8;--shiki-default:#A65E2B;--shiki-dark:#C99076}html pre.shiki code .sa2tF, html code.shiki .sa2tF{--shiki-light:#E2931D;--shiki-default:#998418;--shiki-dark:#B8A965}html pre.shiki code .sJdAF, html code.shiki .sJdAF{--shiki-light:#6182B8;--shiki-default:#998418;--shiki-dark:#B8A965}html pre.shiki code .s131V, html code.shiki .s131V{--shiki-light:#90A4AE;--shiki-default:#998418;--shiki-dark:#B8A965}html pre.shiki code .s3h35, html code.shiki .s3h35{--shiki-light:#F76D47;--shiki-default:#A65E2B;--shiki-dark:#C99076}html pre.shiki code .sFGJz, html code.shiki .sFGJz{--shiki-light:#E53935;--shiki-default:#A65E2B;--shiki-dark:#C99076}html pre.shiki code .sSC40, html code.shiki .sSC40{--shiki-light:#90A4AE;--shiki-default:#B07D48;--shiki-dark:#BD976A}html pre.shiki code .s27EL, html code.shiki .s27EL{--shiki-light:#91B859;--shiki-default:#AB5E3F;--shiki-dark:#C4704F}html pre.shiki code .sRJPX, html code.shiki .sRJPX{--shiki-light:#91B859;--shiki-default:#998418;--shiki-dark:#B8A965}html pre.shiki code .sGuSu, html code.shiki .sGuSu{--shiki-light:#39ADB5;--shiki-default:#2F798A;--shiki-dark:#4C9A91}html pre.shiki code .sETVe, html code.shiki .sETVe{--shiki-light:#39ADB5;--shiki-default:#A65E2B;--shiki-dark:#C99076}html pre.shiki code .sz9Cv, html code.shiki .sz9Cv{--shiki-light:#91B859;--shiki-default:#A65E2B;--shiki-dark:#C99076}",{"title":1045,"searchDepth":1053,"depth":1060,"links":12007},[12008,12013,12018,12022,12026,12030,12033],{"id":1032,"depth":1060,"text":1033,"children":12009},[12010,12011,12012],{"id":1037,"depth":1071,"text":1038},{"id":1408,"depth":1071,"text":1409},{"id":1848,"depth":1071,"text":1849},{"id":2845,"depth":1060,"text":2846,"children":12014},[12015,12016,12017],{"id":2849,"depth":1071,"text":2850},{"id":3218,"depth":1071,"text":3219},{"id":4260,"depth":1071,"text":4261},{"id":5300,"depth":1060,"text":5301,"children":12019},[12020,12021],{"id":5304,"depth":1071,"text":5305},{"id":6045,"depth":1071,"text":6046},{"id":7446,"depth":1060,"text":767,"children":12023},[12024,12025],{"id":7449,"depth":1071,"text":7450},{"id":8894,"depth":1071,"text":8895},{"id":9867,"depth":1060,"text":9868,"children":12027},[12028,12029],{"id":9871,"depth":1071,"text":9872},{"id":10457,"depth":1071,"text":10458},{"id":10780,"depth":1060,"text":10781,"children":12031},[12032],{"id":10784,"depth":1071,"text":10785},{"id":11908,"depth":1060,"text":11909,"children":12034},[12035,12036,12037,12038,12039],{"id":11912,"depth":1071,"text":11913},{"id":11930,"depth":1071,"text":11931},{"id":11948,"depth":1071,"text":11949},{"id":11966,"depth":1071,"text":11967},{"id":11984,"depth":1071,"text":11985},"md",null,{},{"title":511,"description":1028},"yP1nt31n79YSiOxYPG2vTFkJI2VvU3GtmvQ5JtZAuvw",[12046,12048],{"title":507,"path":508,"stem":509,"description":12047,"children":-1},"Admin actions provide a powerful way to perform bulk operations on selected objects in the Django admin interface. This chapter covers creating custom actions, handling complex operations, and building efficient bulk processing tools.",{"title":515,"path":516,"stem":517,"description":12049,"children":-1},"Django middleware is a powerful framework for processing requests and responses globally across your application. It provides hooks into Django's request/response processing, allowing you to modify requests before they reach views and responses before they're sent to clients.",1772474935792]