[{"data":1,"prerenderedAt":12191},["ShallowReactive",2],{"navigation":3,"/middleware/creating-custom-middleware":1016,"/middleware/creating-custom-middleware-surround":12186},[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":529,"body":1018,"description":1028,"extension":12181,"links":12182,"meta":12183,"navigation":1108,"path":530,"seo":12184,"stem":531,"__hash__":12185},"docs/13.middleware/04.creating-custom-middleware.md",{"type":1019,"value":1020,"toc":12150},"minimark",[1021,1025,1029,1034,1039,1482,1486,2336,2340,2344,3632,3636,5122,5126,6406,6410,7157,7161,8171,8175,8179,8908,8912,10147,10151,10155,11087,11091,11506,11510,11514,12062,12066,12070,12086,12090,12104,12107,12121,12125,12139,12143,12146],[1022,1023,529],"h1",{"id":1024},"creating-custom-middleware",[1026,1027,1028],"p",{},"Custom middleware allows you to implement application-specific functionality that runs for every request. This chapter covers designing, implementing, and testing custom middleware for various use cases.",[1030,1031,1033],"h2",{"id":1032},"basic-custom-middleware-structure","Basic Custom Middleware Structure",[1035,1036,1038],"h3",{"id":1037},"simple-middleware-template","Simple Middleware Template",[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","# myapp/middleware.py\nimport time\nimport logging\nfrom django.utils.deprecation import MiddlewareMixin\n\nlogger = logging.getLogger(__name__)\n\nclass BasicCustomMiddleware:\n    \"\"\"Template for custom middleware\"\"\"\n    \n    def __init__(self, get_response):\n        \"\"\"\n        One-time configuration and initialization.\n        \"\"\"\n        self.get_response = get_response\n        # One-time initialization code here\n        logger.info(\"BasicCustomMiddleware initialized\")\n    \n    def __call__(self, request):\n        \"\"\"\n        Code to be executed for each request before the view is called.\n        \"\"\"\n        # Process request\n        start_time = time.time()\n        \n        # Call the next middleware or view\n        response = self.get_response(request)\n        \n        # Process response\n        end_time = time.time()\n        processing_time = end_time - start_time\n        \n        # Add custom header\n        response['X-Processing-Time'] = f\"{processing_time:.4f}\"\n        \n        return response\n","python","",[1047,1048,1049,1058,1069,1077,1103,1110,1138,1143,1157,1171,1177,1203,1209,1215,1220,1239,1245,1270,1275,1294,1299,1305,1310,1316,1335,1341,1347,1369,1374,1380,1396,1414,1419,1425,1468,1473],"code",{"__ignoreMap":1045},[1050,1051,1054],"span",{"class":1052,"line":1053},"line",1,[1050,1055,1057],{"class":1056},"s9Tkl","# myapp/middleware.py\n",[1050,1059,1061,1065],{"class":1052,"line":1060},2,[1050,1062,1064],{"class":1063},"siDh9","import",[1050,1066,1068],{"class":1067},"sftqT"," time\n",[1050,1070,1072,1074],{"class":1052,"line":1071},3,[1050,1073,1064],{"class":1063},[1050,1075,1076],{"class":1067}," logging\n",[1050,1078,1080,1083,1086,1090,1093,1095,1098,1100],{"class":1052,"line":1079},4,[1050,1081,1082],{"class":1063},"from",[1050,1084,1085],{"class":1067}," django",[1050,1087,1089],{"class":1088},"soVBu",".",[1050,1091,1092],{"class":1067},"utils",[1050,1094,1089],{"class":1088},[1050,1096,1097],{"class":1067},"deprecation ",[1050,1099,1064],{"class":1063},[1050,1101,1102],{"class":1067}," MiddlewareMixin\n",[1050,1104,1106],{"class":1052,"line":1105},5,[1050,1107,1109],{"emptyLinePlaceholder":1108},true,"\n",[1050,1111,1113,1116,1119,1122,1124,1128,1131,1135],{"class":1052,"line":1112},6,[1050,1114,1115],{"class":1067},"logger ",[1050,1117,1118],{"class":1088},"=",[1050,1120,1121],{"class":1067}," logging",[1050,1123,1089],{"class":1088},[1050,1125,1127],{"class":1126},"siWMO","getLogger",[1050,1129,1130],{"class":1088},"(",[1050,1132,1134],{"class":1133},"s131V","__name__",[1050,1136,1137],{"class":1088},")\n",[1050,1139,1141],{"class":1052,"line":1140},7,[1050,1142,1109],{"emptyLinePlaceholder":1108},[1050,1144,1146,1150,1154],{"class":1052,"line":1145},8,[1050,1147,1149],{"class":1148},"s5Kfy","class",[1050,1151,1153],{"class":1152},"sD-vU"," BasicCustomMiddleware",[1050,1155,1156],{"class":1088},":\n",[1050,1158,1160,1164,1168],{"class":1052,"line":1159},9,[1050,1161,1163],{"class":1162},"sm7ve","    \"\"\"",[1050,1165,1167],{"class":1166},"sVyVU","Template for custom middleware",[1050,1169,1170],{"class":1162},"\"\"\"\n",[1050,1172,1174],{"class":1052,"line":1173},10,[1050,1175,1176],{"class":1067},"    \n",[1050,1178,1180,1183,1187,1189,1193,1196,1200],{"class":1052,"line":1179},11,[1050,1181,1182],{"class":1148},"    def",[1050,1184,1186],{"class":1185},"sJdAF"," __init__",[1050,1188,1130],{"class":1088},[1050,1190,1192],{"class":1191},"sRjD_","self",[1050,1194,1195],{"class":1088},",",[1050,1197,1199],{"class":1198},"sCyAa"," get_response",[1050,1201,1202],{"class":1088},"):\n",[1050,1204,1206],{"class":1052,"line":1205},12,[1050,1207,1208],{"class":1162},"        \"\"\"\n",[1050,1210,1212],{"class":1052,"line":1211},13,[1050,1213,1214],{"class":1166},"        One-time configuration and initialization.\n",[1050,1216,1218],{"class":1052,"line":1217},14,[1050,1219,1208],{"class":1162},[1050,1221,1223,1227,1229,1233,1236],{"class":1052,"line":1222},15,[1050,1224,1226],{"class":1225},"se3Ec","        self",[1050,1228,1089],{"class":1088},[1050,1230,1232],{"class":1231},"sBPpx","get_response",[1050,1234,1235],{"class":1088}," =",[1050,1237,1238],{"class":1067}," get_response\n",[1050,1240,1242],{"class":1052,"line":1241},16,[1050,1243,1244],{"class":1056},"        # One-time initialization code here\n",[1050,1246,1248,1251,1253,1256,1258,1262,1266,1268],{"class":1052,"line":1247},17,[1050,1249,1250],{"class":1067},"        logger",[1050,1252,1089],{"class":1088},[1050,1254,1255],{"class":1126},"info",[1050,1257,1130],{"class":1088},[1050,1259,1261],{"class":1260},"sbYkP","\"",[1050,1263,1265],{"class":1264},"sTbE_","BasicCustomMiddleware initialized",[1050,1267,1261],{"class":1260},[1050,1269,1137],{"class":1088},[1050,1271,1273],{"class":1052,"line":1272},18,[1050,1274,1176],{"class":1067},[1050,1276,1278,1280,1283,1285,1287,1289,1292],{"class":1052,"line":1277},19,[1050,1279,1182],{"class":1148},[1050,1281,1282],{"class":1185}," __call__",[1050,1284,1130],{"class":1088},[1050,1286,1192],{"class":1191},[1050,1288,1195],{"class":1088},[1050,1290,1291],{"class":1198}," request",[1050,1293,1202],{"class":1088},[1050,1295,1297],{"class":1052,"line":1296},20,[1050,1298,1208],{"class":1162},[1050,1300,1302],{"class":1052,"line":1301},21,[1050,1303,1304],{"class":1166},"        Code to be executed for each request before the view is called.\n",[1050,1306,1308],{"class":1052,"line":1307},22,[1050,1309,1208],{"class":1162},[1050,1311,1313],{"class":1052,"line":1312},23,[1050,1314,1315],{"class":1056},"        # Process request\n",[1050,1317,1319,1322,1324,1327,1329,1332],{"class":1052,"line":1318},24,[1050,1320,1321],{"class":1067},"        start_time ",[1050,1323,1118],{"class":1088},[1050,1325,1326],{"class":1067}," time",[1050,1328,1089],{"class":1088},[1050,1330,1331],{"class":1126},"time",[1050,1333,1334],{"class":1088},"()\n",[1050,1336,1338],{"class":1052,"line":1337},25,[1050,1339,1340],{"class":1067},"        \n",[1050,1342,1344],{"class":1052,"line":1343},26,[1050,1345,1346],{"class":1056},"        # Call the next middleware or view\n",[1050,1348,1350,1353,1355,1358,1360,1362,1364,1367],{"class":1052,"line":1349},27,[1050,1351,1352],{"class":1067},"        response ",[1050,1354,1118],{"class":1088},[1050,1356,1357],{"class":1225}," self",[1050,1359,1089],{"class":1088},[1050,1361,1232],{"class":1126},[1050,1363,1130],{"class":1088},[1050,1365,1366],{"class":1126},"request",[1050,1368,1137],{"class":1088},[1050,1370,1372],{"class":1052,"line":1371},28,[1050,1373,1340],{"class":1067},[1050,1375,1377],{"class":1052,"line":1376},29,[1050,1378,1379],{"class":1056},"        # Process response\n",[1050,1381,1383,1386,1388,1390,1392,1394],{"class":1052,"line":1382},30,[1050,1384,1385],{"class":1067},"        end_time ",[1050,1387,1118],{"class":1088},[1050,1389,1326],{"class":1067},[1050,1391,1089],{"class":1088},[1050,1393,1331],{"class":1126},[1050,1395,1334],{"class":1088},[1050,1397,1399,1402,1404,1407,1411],{"class":1052,"line":1398},31,[1050,1400,1401],{"class":1067},"        processing_time ",[1050,1403,1118],{"class":1088},[1050,1405,1406],{"class":1067}," end_time ",[1050,1408,1410],{"class":1409},"sVsLi","-",[1050,1412,1413],{"class":1067}," start_time\n",[1050,1415,1417],{"class":1052,"line":1416},32,[1050,1418,1340],{"class":1067},[1050,1420,1422],{"class":1052,"line":1421},33,[1050,1423,1424],{"class":1056},"        # Add custom header\n",[1050,1426,1428,1431,1434,1437,1440,1442,1445,1447,1450,1452,1456,1459,1462,1465],{"class":1052,"line":1427},34,[1050,1429,1430],{"class":1067},"        response",[1050,1432,1433],{"class":1088},"[",[1050,1435,1436],{"class":1260},"'",[1050,1438,1439],{"class":1264},"X-Processing-Time",[1050,1441,1436],{"class":1260},[1050,1443,1444],{"class":1088},"]",[1050,1446,1235],{"class":1088},[1050,1448,1449],{"class":1148}," f",[1050,1451,1261],{"class":1264},[1050,1453,1455],{"class":1454},"s3h35","{",[1050,1457,1458],{"class":1067},"processing_time",[1050,1460,1461],{"class":1148},":.4f",[1050,1463,1464],{"class":1454},"}",[1050,1466,1467],{"class":1264},"\"\n",[1050,1469,1471],{"class":1052,"line":1470},35,[1050,1472,1340],{"class":1067},[1050,1474,1476,1479],{"class":1052,"line":1475},36,[1050,1477,1478],{"class":1063},"        return",[1050,1480,1481],{"class":1067}," response\n",[1035,1483,1485],{"id":1484},"middleware-with-all-hooks","Middleware with All Hooks",[1040,1487,1489],{"className":1042,"code":1488,"language":1044,"meta":1045,"style":1045},"class FullFeaturedMiddleware:\n    \"\"\"Middleware demonstrating all available hooks\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n    \n    def __call__(self, request):\n        # This code is executed before the view\n        self.process_request(request)\n        \n        response = self.get_response(request)\n        \n        # This code is executed after the view\n        return self.process_response(request, response)\n    \n    def process_request(self, request):\n        \"\"\"Process the request before it reaches the view\"\"\"\n        # Add custom attributes to request\n        request.custom_timestamp = time.time()\n        request.custom_id = self.generate_request_id()\n        \n        logger.info(f\"Processing request {request.custom_id} to {request.path}\")\n    \n    def process_view(self, request, view_func, view_args, view_kwargs):\n        \"\"\"Called just before Django calls the view\"\"\"\n        logger.info(f\"About to call view: {view_func.__name__}\")\n        \n        # Could return HttpResponse to short-circuit view execution\n        return None\n    \n    def process_exception(self, request, exception):\n        \"\"\"Called when a view raises an exception\"\"\"\n        logger.error(f\"Exception in request {getattr(request, 'custom_id', 'unknown')}: {exception}\")\n        \n        # Could return HttpResponse to handle the exception\n        return None\n    \n    def process_template_response(self, request, response):\n        \"\"\"Called for responses with render() method\"\"\"\n        if hasattr(response, 'context_data'):\n            response.context_data['request_id'] = getattr(request, 'custom_id', None)\n        \n        return response\n    \n    def process_response(self, request, response):\n        \"\"\"Process the response before it's sent to the client\"\"\"\n        # Add custom headers\n        response['X-Request-ID'] = getattr(request, 'custom_id', 'unknown')\n        \n        # Calculate processing time\n        if hasattr(request, 'custom_timestamp'):\n            processing_time = time.time() - request.custom_timestamp\n            response['X-Processing-Time'] = f\"{processing_time:.4f}\"\n        \n        return response\n    \n    def generate_request_id(self):\n        \"\"\"Generate unique request ID\"\"\"\n        import uuid\n        return str(uuid.uuid4())[:8]\n",[1047,1490,1491,1500,1509,1513,1529,1541,1545,1561,1566,1581,1585,1603,1607,1612,1634,1638,1656,1666,1671,1691,1711,1715,1759,1763,1795,1804,1834,1838,1843,1851,1855,1877,1886,1948,1952,1957,1963,1968,1990,2000,2025,2070,2075,2082,2087,2109,2119,2125,2167,2172,2178,2199,2226,2257,2262,2269,2274,2288,2298,2307],{"__ignoreMap":1045},[1050,1492,1493,1495,1498],{"class":1052,"line":1053},[1050,1494,1149],{"class":1148},[1050,1496,1497],{"class":1152}," FullFeaturedMiddleware",[1050,1499,1156],{"class":1088},[1050,1501,1502,1504,1507],{"class":1052,"line":1060},[1050,1503,1163],{"class":1162},[1050,1505,1506],{"class":1166},"Middleware demonstrating all available hooks",[1050,1508,1170],{"class":1162},[1050,1510,1511],{"class":1052,"line":1071},[1050,1512,1176],{"class":1067},[1050,1514,1515,1517,1519,1521,1523,1525,1527],{"class":1052,"line":1079},[1050,1516,1182],{"class":1148},[1050,1518,1186],{"class":1185},[1050,1520,1130],{"class":1088},[1050,1522,1192],{"class":1191},[1050,1524,1195],{"class":1088},[1050,1526,1199],{"class":1198},[1050,1528,1202],{"class":1088},[1050,1530,1531,1533,1535,1537,1539],{"class":1052,"line":1105},[1050,1532,1226],{"class":1225},[1050,1534,1089],{"class":1088},[1050,1536,1232],{"class":1231},[1050,1538,1235],{"class":1088},[1050,1540,1238],{"class":1067},[1050,1542,1543],{"class":1052,"line":1112},[1050,1544,1176],{"class":1067},[1050,1546,1547,1549,1551,1553,1555,1557,1559],{"class":1052,"line":1140},[1050,1548,1182],{"class":1148},[1050,1550,1282],{"class":1185},[1050,1552,1130],{"class":1088},[1050,1554,1192],{"class":1191},[1050,1556,1195],{"class":1088},[1050,1558,1291],{"class":1198},[1050,1560,1202],{"class":1088},[1050,1562,1563],{"class":1052,"line":1145},[1050,1564,1565],{"class":1056},"        # This code is executed before the view\n",[1050,1567,1568,1570,1572,1575,1577,1579],{"class":1052,"line":1159},[1050,1569,1226],{"class":1225},[1050,1571,1089],{"class":1088},[1050,1573,1574],{"class":1126},"process_request",[1050,1576,1130],{"class":1088},[1050,1578,1366],{"class":1126},[1050,1580,1137],{"class":1088},[1050,1582,1583],{"class":1052,"line":1173},[1050,1584,1340],{"class":1067},[1050,1586,1587,1589,1591,1593,1595,1597,1599,1601],{"class":1052,"line":1179},[1050,1588,1352],{"class":1067},[1050,1590,1118],{"class":1088},[1050,1592,1357],{"class":1225},[1050,1594,1089],{"class":1088},[1050,1596,1232],{"class":1126},[1050,1598,1130],{"class":1088},[1050,1600,1366],{"class":1126},[1050,1602,1137],{"class":1088},[1050,1604,1605],{"class":1052,"line":1205},[1050,1606,1340],{"class":1067},[1050,1608,1609],{"class":1052,"line":1211},[1050,1610,1611],{"class":1056},"        # This code is executed after the view\n",[1050,1613,1614,1616,1618,1620,1623,1625,1627,1629,1632],{"class":1052,"line":1217},[1050,1615,1478],{"class":1063},[1050,1617,1357],{"class":1225},[1050,1619,1089],{"class":1088},[1050,1621,1622],{"class":1126},"process_response",[1050,1624,1130],{"class":1088},[1050,1626,1366],{"class":1126},[1050,1628,1195],{"class":1088},[1050,1630,1631],{"class":1126}," response",[1050,1633,1137],{"class":1088},[1050,1635,1636],{"class":1052,"line":1222},[1050,1637,1176],{"class":1067},[1050,1639,1640,1642,1646,1648,1650,1652,1654],{"class":1052,"line":1241},[1050,1641,1182],{"class":1148},[1050,1643,1645],{"class":1644},"sljsM"," process_request",[1050,1647,1130],{"class":1088},[1050,1649,1192],{"class":1191},[1050,1651,1195],{"class":1088},[1050,1653,1291],{"class":1198},[1050,1655,1202],{"class":1088},[1050,1657,1658,1661,1664],{"class":1052,"line":1247},[1050,1659,1660],{"class":1162},"        \"\"\"",[1050,1662,1663],{"class":1166},"Process the request before it reaches the view",[1050,1665,1170],{"class":1162},[1050,1667,1668],{"class":1052,"line":1272},[1050,1669,1670],{"class":1056},"        # Add custom attributes to request\n",[1050,1672,1673,1676,1678,1681,1683,1685,1687,1689],{"class":1052,"line":1277},[1050,1674,1675],{"class":1067},"        request",[1050,1677,1089],{"class":1088},[1050,1679,1680],{"class":1231},"custom_timestamp",[1050,1682,1235],{"class":1088},[1050,1684,1326],{"class":1067},[1050,1686,1089],{"class":1088},[1050,1688,1331],{"class":1126},[1050,1690,1334],{"class":1088},[1050,1692,1693,1695,1697,1700,1702,1704,1706,1709],{"class":1052,"line":1296},[1050,1694,1675],{"class":1067},[1050,1696,1089],{"class":1088},[1050,1698,1699],{"class":1231},"custom_id",[1050,1701,1235],{"class":1088},[1050,1703,1357],{"class":1225},[1050,1705,1089],{"class":1088},[1050,1707,1708],{"class":1126},"generate_request_id",[1050,1710,1334],{"class":1088},[1050,1712,1713],{"class":1052,"line":1301},[1050,1714,1340],{"class":1067},[1050,1716,1717,1719,1721,1723,1725,1728,1731,1733,1735,1737,1739,1741,1744,1746,1748,1750,1753,1755,1757],{"class":1052,"line":1307},[1050,1718,1250],{"class":1067},[1050,1720,1089],{"class":1088},[1050,1722,1255],{"class":1126},[1050,1724,1130],{"class":1088},[1050,1726,1727],{"class":1148},"f",[1050,1729,1730],{"class":1264},"\"Processing request ",[1050,1732,1455],{"class":1454},[1050,1734,1366],{"class":1126},[1050,1736,1089],{"class":1088},[1050,1738,1699],{"class":1231},[1050,1740,1464],{"class":1454},[1050,1742,1743],{"class":1264}," to ",[1050,1745,1455],{"class":1454},[1050,1747,1366],{"class":1126},[1050,1749,1089],{"class":1088},[1050,1751,1752],{"class":1231},"path",[1050,1754,1464],{"class":1454},[1050,1756,1261],{"class":1264},[1050,1758,1137],{"class":1088},[1050,1760,1761],{"class":1052,"line":1312},[1050,1762,1176],{"class":1067},[1050,1764,1765,1767,1770,1772,1774,1776,1778,1780,1783,1785,1788,1790,1793],{"class":1052,"line":1318},[1050,1766,1182],{"class":1148},[1050,1768,1769],{"class":1644}," process_view",[1050,1771,1130],{"class":1088},[1050,1773,1192],{"class":1191},[1050,1775,1195],{"class":1088},[1050,1777,1291],{"class":1198},[1050,1779,1195],{"class":1088},[1050,1781,1782],{"class":1198}," view_func",[1050,1784,1195],{"class":1088},[1050,1786,1787],{"class":1198}," view_args",[1050,1789,1195],{"class":1088},[1050,1791,1792],{"class":1198}," view_kwargs",[1050,1794,1202],{"class":1088},[1050,1796,1797,1799,1802],{"class":1052,"line":1337},[1050,1798,1660],{"class":1162},[1050,1800,1801],{"class":1166},"Called just before Django calls the view",[1050,1803,1170],{"class":1162},[1050,1805,1806,1808,1810,1812,1814,1816,1819,1821,1824,1826,1828,1830,1832],{"class":1052,"line":1343},[1050,1807,1250],{"class":1067},[1050,1809,1089],{"class":1088},[1050,1811,1255],{"class":1126},[1050,1813,1130],{"class":1088},[1050,1815,1727],{"class":1148},[1050,1817,1818],{"class":1264},"\"About to call view: ",[1050,1820,1455],{"class":1454},[1050,1822,1823],{"class":1126},"view_func",[1050,1825,1089],{"class":1088},[1050,1827,1134],{"class":1133},[1050,1829,1464],{"class":1454},[1050,1831,1261],{"class":1264},[1050,1833,1137],{"class":1088},[1050,1835,1836],{"class":1052,"line":1349},[1050,1837,1340],{"class":1067},[1050,1839,1840],{"class":1052,"line":1371},[1050,1841,1842],{"class":1056},"        # Could return HttpResponse to short-circuit view execution\n",[1050,1844,1845,1847],{"class":1052,"line":1376},[1050,1846,1478],{"class":1063},[1050,1848,1850],{"class":1849},"s8XtY"," None\n",[1050,1852,1853],{"class":1052,"line":1382},[1050,1854,1176],{"class":1067},[1050,1856,1857,1859,1862,1864,1866,1868,1870,1872,1875],{"class":1052,"line":1398},[1050,1858,1182],{"class":1148},[1050,1860,1861],{"class":1644}," process_exception",[1050,1863,1130],{"class":1088},[1050,1865,1192],{"class":1191},[1050,1867,1195],{"class":1088},[1050,1869,1291],{"class":1198},[1050,1871,1195],{"class":1088},[1050,1873,1874],{"class":1198}," exception",[1050,1876,1202],{"class":1088},[1050,1878,1879,1881,1884],{"class":1052,"line":1416},[1050,1880,1660],{"class":1162},[1050,1882,1883],{"class":1166},"Called when a view raises an exception",[1050,1885,1170],{"class":1162},[1050,1887,1888,1890,1892,1895,1897,1899,1902,1904,1907,1909,1911,1913,1916,1918,1920,1922,1924,1927,1929,1932,1934,1937,1939,1942,1944,1946],{"class":1052,"line":1421},[1050,1889,1250],{"class":1067},[1050,1891,1089],{"class":1088},[1050,1893,1894],{"class":1126},"error",[1050,1896,1130],{"class":1088},[1050,1898,1727],{"class":1148},[1050,1900,1901],{"class":1264},"\"Exception in request ",[1050,1903,1455],{"class":1454},[1050,1905,1906],{"class":1185},"getattr",[1050,1908,1130],{"class":1088},[1050,1910,1366],{"class":1126},[1050,1912,1195],{"class":1088},[1050,1914,1915],{"class":1260}," '",[1050,1917,1699],{"class":1264},[1050,1919,1436],{"class":1260},[1050,1921,1195],{"class":1088},[1050,1923,1915],{"class":1260},[1050,1925,1926],{"class":1264},"unknown",[1050,1928,1436],{"class":1260},[1050,1930,1931],{"class":1088},")",[1050,1933,1464],{"class":1454},[1050,1935,1936],{"class":1264},": ",[1050,1938,1455],{"class":1454},[1050,1940,1941],{"class":1126},"exception",[1050,1943,1464],{"class":1454},[1050,1945,1261],{"class":1264},[1050,1947,1137],{"class":1088},[1050,1949,1950],{"class":1052,"line":1427},[1050,1951,1340],{"class":1067},[1050,1953,1954],{"class":1052,"line":1470},[1050,1955,1956],{"class":1056},"        # Could return HttpResponse to handle the exception\n",[1050,1958,1959,1961],{"class":1052,"line":1475},[1050,1960,1478],{"class":1063},[1050,1962,1850],{"class":1849},[1050,1964,1966],{"class":1052,"line":1965},37,[1050,1967,1176],{"class":1067},[1050,1969,1971,1973,1976,1978,1980,1982,1984,1986,1988],{"class":1052,"line":1970},38,[1050,1972,1182],{"class":1148},[1050,1974,1975],{"class":1644}," process_template_response",[1050,1977,1130],{"class":1088},[1050,1979,1192],{"class":1191},[1050,1981,1195],{"class":1088},[1050,1983,1291],{"class":1198},[1050,1985,1195],{"class":1088},[1050,1987,1631],{"class":1198},[1050,1989,1202],{"class":1088},[1050,1991,1993,1995,1998],{"class":1052,"line":1992},39,[1050,1994,1660],{"class":1162},[1050,1996,1997],{"class":1166},"Called for responses with render() method",[1050,1999,1170],{"class":1162},[1050,2001,2003,2006,2009,2011,2014,2016,2018,2021,2023],{"class":1052,"line":2002},40,[1050,2004,2005],{"class":1063},"        if",[1050,2007,2008],{"class":1185}," hasattr",[1050,2010,1130],{"class":1088},[1050,2012,2013],{"class":1126},"response",[1050,2015,1195],{"class":1088},[1050,2017,1915],{"class":1260},[1050,2019,2020],{"class":1264},"context_data",[1050,2022,1436],{"class":1260},[1050,2024,1202],{"class":1088},[1050,2026,2028,2031,2033,2035,2037,2039,2042,2044,2046,2048,2051,2053,2055,2057,2059,2061,2063,2065,2068],{"class":1052,"line":2027},41,[1050,2029,2030],{"class":1067},"            response",[1050,2032,1089],{"class":1088},[1050,2034,2020],{"class":1231},[1050,2036,1433],{"class":1088},[1050,2038,1436],{"class":1260},[1050,2040,2041],{"class":1264},"request_id",[1050,2043,1436],{"class":1260},[1050,2045,1444],{"class":1088},[1050,2047,1235],{"class":1088},[1050,2049,2050],{"class":1185}," getattr",[1050,2052,1130],{"class":1088},[1050,2054,1366],{"class":1126},[1050,2056,1195],{"class":1088},[1050,2058,1915],{"class":1260},[1050,2060,1699],{"class":1264},[1050,2062,1436],{"class":1260},[1050,2064,1195],{"class":1088},[1050,2066,2067],{"class":1849}," None",[1050,2069,1137],{"class":1088},[1050,2071,2073],{"class":1052,"line":2072},42,[1050,2074,1340],{"class":1067},[1050,2076,2078,2080],{"class":1052,"line":2077},43,[1050,2079,1478],{"class":1063},[1050,2081,1481],{"class":1067},[1050,2083,2085],{"class":1052,"line":2084},44,[1050,2086,1176],{"class":1067},[1050,2088,2090,2092,2095,2097,2099,2101,2103,2105,2107],{"class":1052,"line":2089},45,[1050,2091,1182],{"class":1148},[1050,2093,2094],{"class":1644}," process_response",[1050,2096,1130],{"class":1088},[1050,2098,1192],{"class":1191},[1050,2100,1195],{"class":1088},[1050,2102,1291],{"class":1198},[1050,2104,1195],{"class":1088},[1050,2106,1631],{"class":1198},[1050,2108,1202],{"class":1088},[1050,2110,2112,2114,2117],{"class":1052,"line":2111},46,[1050,2113,1660],{"class":1162},[1050,2115,2116],{"class":1166},"Process the response before it's sent to the client",[1050,2118,1170],{"class":1162},[1050,2120,2122],{"class":1052,"line":2121},47,[1050,2123,2124],{"class":1056},"        # Add custom headers\n",[1050,2126,2128,2130,2132,2134,2137,2139,2141,2143,2145,2147,2149,2151,2153,2155,2157,2159,2161,2163,2165],{"class":1052,"line":2127},48,[1050,2129,1430],{"class":1067},[1050,2131,1433],{"class":1088},[1050,2133,1436],{"class":1260},[1050,2135,2136],{"class":1264},"X-Request-ID",[1050,2138,1436],{"class":1260},[1050,2140,1444],{"class":1088},[1050,2142,1235],{"class":1088},[1050,2144,2050],{"class":1185},[1050,2146,1130],{"class":1088},[1050,2148,1366],{"class":1126},[1050,2150,1195],{"class":1088},[1050,2152,1915],{"class":1260},[1050,2154,1699],{"class":1264},[1050,2156,1436],{"class":1260},[1050,2158,1195],{"class":1088},[1050,2160,1915],{"class":1260},[1050,2162,1926],{"class":1264},[1050,2164,1436],{"class":1260},[1050,2166,1137],{"class":1088},[1050,2168,2170],{"class":1052,"line":2169},49,[1050,2171,1340],{"class":1067},[1050,2173,2175],{"class":1052,"line":2174},50,[1050,2176,2177],{"class":1056},"        # Calculate processing time\n",[1050,2179,2181,2183,2185,2187,2189,2191,2193,2195,2197],{"class":1052,"line":2180},51,[1050,2182,2005],{"class":1063},[1050,2184,2008],{"class":1185},[1050,2186,1130],{"class":1088},[1050,2188,1366],{"class":1126},[1050,2190,1195],{"class":1088},[1050,2192,1915],{"class":1260},[1050,2194,1680],{"class":1264},[1050,2196,1436],{"class":1260},[1050,2198,1202],{"class":1088},[1050,2200,2202,2205,2207,2209,2211,2213,2216,2219,2221,2223],{"class":1052,"line":2201},52,[1050,2203,2204],{"class":1067},"            processing_time ",[1050,2206,1118],{"class":1088},[1050,2208,1326],{"class":1067},[1050,2210,1089],{"class":1088},[1050,2212,1331],{"class":1126},[1050,2214,2215],{"class":1088},"()",[1050,2217,2218],{"class":1409}," -",[1050,2220,1291],{"class":1067},[1050,2222,1089],{"class":1088},[1050,2224,2225],{"class":1231},"custom_timestamp\n",[1050,2227,2229,2231,2233,2235,2237,2239,2241,2243,2245,2247,2249,2251,2253,2255],{"class":1052,"line":2228},53,[1050,2230,2030],{"class":1067},[1050,2232,1433],{"class":1088},[1050,2234,1436],{"class":1260},[1050,2236,1439],{"class":1264},[1050,2238,1436],{"class":1260},[1050,2240,1444],{"class":1088},[1050,2242,1235],{"class":1088},[1050,2244,1449],{"class":1148},[1050,2246,1261],{"class":1264},[1050,2248,1455],{"class":1454},[1050,2250,1458],{"class":1067},[1050,2252,1461],{"class":1148},[1050,2254,1464],{"class":1454},[1050,2256,1467],{"class":1264},[1050,2258,2260],{"class":1052,"line":2259},54,[1050,2261,1340],{"class":1067},[1050,2263,2265,2267],{"class":1052,"line":2264},55,[1050,2266,1478],{"class":1063},[1050,2268,1481],{"class":1067},[1050,2270,2272],{"class":1052,"line":2271},56,[1050,2273,1176],{"class":1067},[1050,2275,2277,2279,2282,2284,2286],{"class":1052,"line":2276},57,[1050,2278,1182],{"class":1148},[1050,2280,2281],{"class":1644}," generate_request_id",[1050,2283,1130],{"class":1088},[1050,2285,1192],{"class":1191},[1050,2287,1202],{"class":1088},[1050,2289,2291,2293,2296],{"class":1052,"line":2290},58,[1050,2292,1660],{"class":1162},[1050,2294,2295],{"class":1166},"Generate unique request ID",[1050,2297,1170],{"class":1162},[1050,2299,2301,2304],{"class":1052,"line":2300},59,[1050,2302,2303],{"class":1063},"        import",[1050,2305,2306],{"class":1067}," uuid\n",[1050,2308,2310,2312,2316,2318,2321,2323,2326,2329,2333],{"class":1052,"line":2309},60,[1050,2311,1478],{"class":1063},[1050,2313,2315],{"class":2314},"sa2tF"," str",[1050,2317,1130],{"class":1088},[1050,2319,2320],{"class":1126},"uuid",[1050,2322,1089],{"class":1088},[1050,2324,2325],{"class":1126},"uuid4",[1050,2327,2328],{"class":1088},"())[:",[1050,2330,2332],{"class":2331},"s7CZa","8",[1050,2334,2335],{"class":1088},"]\n",[1030,2337,2339],{"id":2338},"practical-middleware-examples","Practical Middleware Examples",[1035,2341,2343],{"id":2342},"request-logging-middleware","Request Logging Middleware",[1040,2345,2347],{"className":1042,"code":2346,"language":1044,"meta":1045,"style":1045},"import json\nimport logging\nfrom django.utils import timezone\n\nlogger = logging.getLogger('request_logger')\n\nclass RequestLoggingMiddleware:\n    \"\"\"Comprehensive request logging middleware\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n        self.sensitive_headers = [\n            'HTTP_AUTHORIZATION',\n            'HTTP_COOKIE',\n            'HTTP_X_API_KEY'\n        ]\n    \n    def __call__(self, request):\n        # Log request\n        request_data = self.extract_request_data(request)\n        logger.info(f\"REQUEST: {json.dumps(request_data)}\")\n        \n        # Process request\n        start_time = time.time()\n        response = self.get_response(request)\n        end_time = time.time()\n        \n        # Log response\n        response_data = self.extract_response_data(request, response, end_time - start_time)\n        logger.info(f\"RESPONSE: {json.dumps(response_data)}\")\n        \n        return response\n    \n    def extract_request_data(self, request):\n        \"\"\"Extract relevant request data for logging\"\"\"\n        return {\n            'timestamp': timezone.now().isoformat(),\n            'method': request.method,\n            'path': request.path,\n            'query_params': dict(request.GET),\n            'user': str(request.user) if hasattr(request, 'user') else 'Anonymous',\n            'ip_address': self.get_client_ip(request),\n            'user_agent': request.META.get('HTTP_USER_AGENT', ''),\n            'content_type': request.META.get('CONTENT_TYPE', ''),\n            'content_length': request.META.get('CONTENT_LENGTH', 0),\n            'headers': self.get_safe_headers(request)\n        }\n    \n    def extract_response_data(self, request, response, processing_time):\n        \"\"\"Extract relevant response data for logging\"\"\"\n        return {\n            'timestamp': timezone.now().isoformat(),\n            'status_code': response.status_code,\n            'content_type': response.get('Content-Type', ''),\n            'content_length': len(response.content) if hasattr(response, 'content') else 0,\n            'processing_time': round(processing_time, 4),\n            'path': request.path\n        }\n    \n    def get_client_ip(self, request):\n        \"\"\"Get the client's IP address\"\"\"\n        x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')\n        if x_forwarded_for:\n            ip = x_forwarded_for.split(',')[0]\n        else:\n            ip = request.META.get('REMOTE_ADDR')\n        return ip\n    \n    def get_safe_headers(self, request):\n        \"\"\"Get headers excluding sensitive information\"\"\"\n        headers = {}\n        for key, value in request.META.items():\n            if key.startswith('HTTP_') and key not in self.sensitive_headers:\n                header_name = key[5:].replace('_', '-').title()\n                headers[header_name] = value\n        return headers\n",[1047,2348,2349,2356,2362,2378,2382,2405,2409,2418,2427,2431,2447,2459,2473,2486,2497,2507,2512,2516,2532,2537,2557,2595,2599,2603,2617,2635,2649,2653,2658,2691,2727,2731,2737,2741,2758,2767,2774,2803,2822,2840,2867,2921,2945,2984,3020,3057,3081,3086,3090,3116,3125,3131,3153,3172,3203,3251,3275,3292,3296,3300,3317,3327,3356,3366,3397,3405,3433,3441,3446,3464,3474,3485,3516,3560,3606,3624],{"__ignoreMap":1045},[1050,2350,2351,2353],{"class":1052,"line":1053},[1050,2352,1064],{"class":1063},[1050,2354,2355],{"class":1067}," json\n",[1050,2357,2358,2360],{"class":1052,"line":1060},[1050,2359,1064],{"class":1063},[1050,2361,1076],{"class":1067},[1050,2363,2364,2366,2368,2370,2373,2375],{"class":1052,"line":1071},[1050,2365,1082],{"class":1063},[1050,2367,1085],{"class":1067},[1050,2369,1089],{"class":1088},[1050,2371,2372],{"class":1067},"utils ",[1050,2374,1064],{"class":1063},[1050,2376,2377],{"class":1067}," timezone\n",[1050,2379,2380],{"class":1052,"line":1079},[1050,2381,1109],{"emptyLinePlaceholder":1108},[1050,2383,2384,2386,2388,2390,2392,2394,2396,2398,2401,2403],{"class":1052,"line":1105},[1050,2385,1115],{"class":1067},[1050,2387,1118],{"class":1088},[1050,2389,1121],{"class":1067},[1050,2391,1089],{"class":1088},[1050,2393,1127],{"class":1126},[1050,2395,1130],{"class":1088},[1050,2397,1436],{"class":1260},[1050,2399,2400],{"class":1264},"request_logger",[1050,2402,1436],{"class":1260},[1050,2404,1137],{"class":1088},[1050,2406,2407],{"class":1052,"line":1112},[1050,2408,1109],{"emptyLinePlaceholder":1108},[1050,2410,2411,2413,2416],{"class":1052,"line":1140},[1050,2412,1149],{"class":1148},[1050,2414,2415],{"class":1152}," RequestLoggingMiddleware",[1050,2417,1156],{"class":1088},[1050,2419,2420,2422,2425],{"class":1052,"line":1145},[1050,2421,1163],{"class":1162},[1050,2423,2424],{"class":1166},"Comprehensive request logging middleware",[1050,2426,1170],{"class":1162},[1050,2428,2429],{"class":1052,"line":1159},[1050,2430,1176],{"class":1067},[1050,2432,2433,2435,2437,2439,2441,2443,2445],{"class":1052,"line":1173},[1050,2434,1182],{"class":1148},[1050,2436,1186],{"class":1185},[1050,2438,1130],{"class":1088},[1050,2440,1192],{"class":1191},[1050,2442,1195],{"class":1088},[1050,2444,1199],{"class":1198},[1050,2446,1202],{"class":1088},[1050,2448,2449,2451,2453,2455,2457],{"class":1052,"line":1179},[1050,2450,1226],{"class":1225},[1050,2452,1089],{"class":1088},[1050,2454,1232],{"class":1231},[1050,2456,1235],{"class":1088},[1050,2458,1238],{"class":1067},[1050,2460,2461,2463,2465,2468,2470],{"class":1052,"line":1205},[1050,2462,1226],{"class":1225},[1050,2464,1089],{"class":1088},[1050,2466,2467],{"class":1231},"sensitive_headers",[1050,2469,1235],{"class":1088},[1050,2471,2472],{"class":1088}," [\n",[1050,2474,2475,2478,2481,2483],{"class":1052,"line":1211},[1050,2476,2477],{"class":1260},"            '",[1050,2479,2480],{"class":1264},"HTTP_AUTHORIZATION",[1050,2482,1436],{"class":1260},[1050,2484,2485],{"class":1088},",\n",[1050,2487,2488,2490,2493,2495],{"class":1052,"line":1217},[1050,2489,2477],{"class":1260},[1050,2491,2492],{"class":1264},"HTTP_COOKIE",[1050,2494,1436],{"class":1260},[1050,2496,2485],{"class":1088},[1050,2498,2499,2501,2504],{"class":1052,"line":1222},[1050,2500,2477],{"class":1260},[1050,2502,2503],{"class":1264},"HTTP_X_API_KEY",[1050,2505,2506],{"class":1260},"'\n",[1050,2508,2509],{"class":1052,"line":1241},[1050,2510,2511],{"class":1088},"        ]\n",[1050,2513,2514],{"class":1052,"line":1247},[1050,2515,1176],{"class":1067},[1050,2517,2518,2520,2522,2524,2526,2528,2530],{"class":1052,"line":1272},[1050,2519,1182],{"class":1148},[1050,2521,1282],{"class":1185},[1050,2523,1130],{"class":1088},[1050,2525,1192],{"class":1191},[1050,2527,1195],{"class":1088},[1050,2529,1291],{"class":1198},[1050,2531,1202],{"class":1088},[1050,2533,2534],{"class":1052,"line":1277},[1050,2535,2536],{"class":1056},"        # Log request\n",[1050,2538,2539,2542,2544,2546,2548,2551,2553,2555],{"class":1052,"line":1296},[1050,2540,2541],{"class":1067},"        request_data ",[1050,2543,1118],{"class":1088},[1050,2545,1357],{"class":1225},[1050,2547,1089],{"class":1088},[1050,2549,2550],{"class":1126},"extract_request_data",[1050,2552,1130],{"class":1088},[1050,2554,1366],{"class":1126},[1050,2556,1137],{"class":1088},[1050,2558,2559,2561,2563,2565,2567,2569,2572,2574,2577,2579,2582,2584,2587,2589,2591,2593],{"class":1052,"line":1301},[1050,2560,1250],{"class":1067},[1050,2562,1089],{"class":1088},[1050,2564,1255],{"class":1126},[1050,2566,1130],{"class":1088},[1050,2568,1727],{"class":1148},[1050,2570,2571],{"class":1264},"\"REQUEST: ",[1050,2573,1455],{"class":1454},[1050,2575,2576],{"class":1126},"json",[1050,2578,1089],{"class":1088},[1050,2580,2581],{"class":1126},"dumps",[1050,2583,1130],{"class":1088},[1050,2585,2586],{"class":1126},"request_data",[1050,2588,1931],{"class":1088},[1050,2590,1464],{"class":1454},[1050,2592,1261],{"class":1264},[1050,2594,1137],{"class":1088},[1050,2596,2597],{"class":1052,"line":1307},[1050,2598,1340],{"class":1067},[1050,2600,2601],{"class":1052,"line":1312},[1050,2602,1315],{"class":1056},[1050,2604,2605,2607,2609,2611,2613,2615],{"class":1052,"line":1318},[1050,2606,1321],{"class":1067},[1050,2608,1118],{"class":1088},[1050,2610,1326],{"class":1067},[1050,2612,1089],{"class":1088},[1050,2614,1331],{"class":1126},[1050,2616,1334],{"class":1088},[1050,2618,2619,2621,2623,2625,2627,2629,2631,2633],{"class":1052,"line":1337},[1050,2620,1352],{"class":1067},[1050,2622,1118],{"class":1088},[1050,2624,1357],{"class":1225},[1050,2626,1089],{"class":1088},[1050,2628,1232],{"class":1126},[1050,2630,1130],{"class":1088},[1050,2632,1366],{"class":1126},[1050,2634,1137],{"class":1088},[1050,2636,2637,2639,2641,2643,2645,2647],{"class":1052,"line":1343},[1050,2638,1385],{"class":1067},[1050,2640,1118],{"class":1088},[1050,2642,1326],{"class":1067},[1050,2644,1089],{"class":1088},[1050,2646,1331],{"class":1126},[1050,2648,1334],{"class":1088},[1050,2650,2651],{"class":1052,"line":1349},[1050,2652,1340],{"class":1067},[1050,2654,2655],{"class":1052,"line":1371},[1050,2656,2657],{"class":1056},"        # Log response\n",[1050,2659,2660,2663,2665,2667,2669,2672,2674,2676,2678,2680,2682,2684,2686,2689],{"class":1052,"line":1376},[1050,2661,2662],{"class":1067},"        response_data ",[1050,2664,1118],{"class":1088},[1050,2666,1357],{"class":1225},[1050,2668,1089],{"class":1088},[1050,2670,2671],{"class":1126},"extract_response_data",[1050,2673,1130],{"class":1088},[1050,2675,1366],{"class":1126},[1050,2677,1195],{"class":1088},[1050,2679,1631],{"class":1126},[1050,2681,1195],{"class":1088},[1050,2683,1406],{"class":1126},[1050,2685,1410],{"class":1409},[1050,2687,2688],{"class":1126}," start_time",[1050,2690,1137],{"class":1088},[1050,2692,2693,2695,2697,2699,2701,2703,2706,2708,2710,2712,2714,2716,2719,2721,2723,2725],{"class":1052,"line":1382},[1050,2694,1250],{"class":1067},[1050,2696,1089],{"class":1088},[1050,2698,1255],{"class":1126},[1050,2700,1130],{"class":1088},[1050,2702,1727],{"class":1148},[1050,2704,2705],{"class":1264},"\"RESPONSE: ",[1050,2707,1455],{"class":1454},[1050,2709,2576],{"class":1126},[1050,2711,1089],{"class":1088},[1050,2713,2581],{"class":1126},[1050,2715,1130],{"class":1088},[1050,2717,2718],{"class":1126},"response_data",[1050,2720,1931],{"class":1088},[1050,2722,1464],{"class":1454},[1050,2724,1261],{"class":1264},[1050,2726,1137],{"class":1088},[1050,2728,2729],{"class":1052,"line":1398},[1050,2730,1340],{"class":1067},[1050,2732,2733,2735],{"class":1052,"line":1416},[1050,2734,1478],{"class":1063},[1050,2736,1481],{"class":1067},[1050,2738,2739],{"class":1052,"line":1421},[1050,2740,1176],{"class":1067},[1050,2742,2743,2745,2748,2750,2752,2754,2756],{"class":1052,"line":1427},[1050,2744,1182],{"class":1148},[1050,2746,2747],{"class":1644}," extract_request_data",[1050,2749,1130],{"class":1088},[1050,2751,1192],{"class":1191},[1050,2753,1195],{"class":1088},[1050,2755,1291],{"class":1198},[1050,2757,1202],{"class":1088},[1050,2759,2760,2762,2765],{"class":1052,"line":1470},[1050,2761,1660],{"class":1162},[1050,2763,2764],{"class":1166},"Extract relevant request data for logging",[1050,2766,1170],{"class":1162},[1050,2768,2769,2771],{"class":1052,"line":1475},[1050,2770,1478],{"class":1063},[1050,2772,2773],{"class":1088}," {\n",[1050,2775,2776,2778,2781,2783,2786,2789,2791,2794,2797,2800],{"class":1052,"line":1965},[1050,2777,2477],{"class":1260},[1050,2779,2780],{"class":1264},"timestamp",[1050,2782,1436],{"class":1260},[1050,2784,2785],{"class":1088},":",[1050,2787,2788],{"class":1067}," timezone",[1050,2790,1089],{"class":1088},[1050,2792,2793],{"class":1126},"now",[1050,2795,2796],{"class":1088},"().",[1050,2798,2799],{"class":1126},"isoformat",[1050,2801,2802],{"class":1088},"(),\n",[1050,2804,2805,2807,2810,2812,2814,2816,2818,2820],{"class":1052,"line":1970},[1050,2806,2477],{"class":1260},[1050,2808,2809],{"class":1264},"method",[1050,2811,1436],{"class":1260},[1050,2813,2785],{"class":1088},[1050,2815,1291],{"class":1067},[1050,2817,1089],{"class":1088},[1050,2819,2809],{"class":1231},[1050,2821,2485],{"class":1088},[1050,2823,2824,2826,2828,2830,2832,2834,2836,2838],{"class":1052,"line":1992},[1050,2825,2477],{"class":1260},[1050,2827,1752],{"class":1264},[1050,2829,1436],{"class":1260},[1050,2831,2785],{"class":1088},[1050,2833,1291],{"class":1067},[1050,2835,1089],{"class":1088},[1050,2837,1752],{"class":1231},[1050,2839,2485],{"class":1088},[1050,2841,2842,2844,2847,2849,2851,2854,2856,2858,2860,2864],{"class":1052,"line":2002},[1050,2843,2477],{"class":1260},[1050,2845,2846],{"class":1264},"query_params",[1050,2848,1436],{"class":1260},[1050,2850,2785],{"class":1088},[1050,2852,2853],{"class":2314}," dict",[1050,2855,1130],{"class":1088},[1050,2857,1366],{"class":1126},[1050,2859,1089],{"class":1088},[1050,2861,2863],{"class":2862},"sFGJz","GET",[1050,2865,2866],{"class":1088},"),\n",[1050,2868,2869,2871,2874,2876,2878,2880,2882,2884,2886,2888,2890,2893,2895,2897,2899,2901,2903,2905,2907,2909,2912,2914,2917,2919],{"class":1052,"line":2027},[1050,2870,2477],{"class":1260},[1050,2872,2873],{"class":1264},"user",[1050,2875,1436],{"class":1260},[1050,2877,2785],{"class":1088},[1050,2879,2315],{"class":2314},[1050,2881,1130],{"class":1088},[1050,2883,1366],{"class":1126},[1050,2885,1089],{"class":1088},[1050,2887,2873],{"class":1231},[1050,2889,1931],{"class":1088},[1050,2891,2892],{"class":1063}," if",[1050,2894,2008],{"class":1185},[1050,2896,1130],{"class":1088},[1050,2898,1366],{"class":1126},[1050,2900,1195],{"class":1088},[1050,2902,1915],{"class":1260},[1050,2904,2873],{"class":1264},[1050,2906,1436],{"class":1260},[1050,2908,1931],{"class":1088},[1050,2910,2911],{"class":1063}," else",[1050,2913,1915],{"class":1260},[1050,2915,2916],{"class":1264},"Anonymous",[1050,2918,1436],{"class":1260},[1050,2920,2485],{"class":1088},[1050,2922,2923,2925,2928,2930,2932,2934,2936,2939,2941,2943],{"class":1052,"line":2072},[1050,2924,2477],{"class":1260},[1050,2926,2927],{"class":1264},"ip_address",[1050,2929,1436],{"class":1260},[1050,2931,2785],{"class":1088},[1050,2933,1357],{"class":1225},[1050,2935,1089],{"class":1088},[1050,2937,2938],{"class":1126},"get_client_ip",[1050,2940,1130],{"class":1088},[1050,2942,1366],{"class":1126},[1050,2944,2866],{"class":1088},[1050,2946,2947,2949,2952,2954,2956,2958,2960,2963,2965,2968,2970,2972,2975,2977,2979,2982],{"class":1052,"line":2077},[1050,2948,2477],{"class":1260},[1050,2950,2951],{"class":1264},"user_agent",[1050,2953,1436],{"class":1260},[1050,2955,2785],{"class":1088},[1050,2957,1291],{"class":1067},[1050,2959,1089],{"class":1088},[1050,2961,2962],{"class":2862},"META",[1050,2964,1089],{"class":1088},[1050,2966,2967],{"class":1126},"get",[1050,2969,1130],{"class":1088},[1050,2971,1436],{"class":1260},[1050,2973,2974],{"class":1264},"HTTP_USER_AGENT",[1050,2976,1436],{"class":1260},[1050,2978,1195],{"class":1088},[1050,2980,2981],{"class":1260}," ''",[1050,2983,2866],{"class":1088},[1050,2985,2986,2988,2991,2993,2995,2997,2999,3001,3003,3005,3007,3009,3012,3014,3016,3018],{"class":1052,"line":2084},[1050,2987,2477],{"class":1260},[1050,2989,2990],{"class":1264},"content_type",[1050,2992,1436],{"class":1260},[1050,2994,2785],{"class":1088},[1050,2996,1291],{"class":1067},[1050,2998,1089],{"class":1088},[1050,3000,2962],{"class":2862},[1050,3002,1089],{"class":1088},[1050,3004,2967],{"class":1126},[1050,3006,1130],{"class":1088},[1050,3008,1436],{"class":1260},[1050,3010,3011],{"class":1264},"CONTENT_TYPE",[1050,3013,1436],{"class":1260},[1050,3015,1195],{"class":1088},[1050,3017,2981],{"class":1260},[1050,3019,2866],{"class":1088},[1050,3021,3022,3024,3027,3029,3031,3033,3035,3037,3039,3041,3043,3045,3048,3050,3052,3055],{"class":1052,"line":2089},[1050,3023,2477],{"class":1260},[1050,3025,3026],{"class":1264},"content_length",[1050,3028,1436],{"class":1260},[1050,3030,2785],{"class":1088},[1050,3032,1291],{"class":1067},[1050,3034,1089],{"class":1088},[1050,3036,2962],{"class":2862},[1050,3038,1089],{"class":1088},[1050,3040,2967],{"class":1126},[1050,3042,1130],{"class":1088},[1050,3044,1436],{"class":1260},[1050,3046,3047],{"class":1264},"CONTENT_LENGTH",[1050,3049,1436],{"class":1260},[1050,3051,1195],{"class":1088},[1050,3053,3054],{"class":2331}," 0",[1050,3056,2866],{"class":1088},[1050,3058,3059,3061,3064,3066,3068,3070,3072,3075,3077,3079],{"class":1052,"line":2111},[1050,3060,2477],{"class":1260},[1050,3062,3063],{"class":1264},"headers",[1050,3065,1436],{"class":1260},[1050,3067,2785],{"class":1088},[1050,3069,1357],{"class":1225},[1050,3071,1089],{"class":1088},[1050,3073,3074],{"class":1126},"get_safe_headers",[1050,3076,1130],{"class":1088},[1050,3078,1366],{"class":1126},[1050,3080,1137],{"class":1088},[1050,3082,3083],{"class":1052,"line":2121},[1050,3084,3085],{"class":1088},"        }\n",[1050,3087,3088],{"class":1052,"line":2127},[1050,3089,1176],{"class":1067},[1050,3091,3092,3094,3097,3099,3101,3103,3105,3107,3109,3111,3114],{"class":1052,"line":2169},[1050,3093,1182],{"class":1148},[1050,3095,3096],{"class":1644}," extract_response_data",[1050,3098,1130],{"class":1088},[1050,3100,1192],{"class":1191},[1050,3102,1195],{"class":1088},[1050,3104,1291],{"class":1198},[1050,3106,1195],{"class":1088},[1050,3108,1631],{"class":1198},[1050,3110,1195],{"class":1088},[1050,3112,3113],{"class":1198}," processing_time",[1050,3115,1202],{"class":1088},[1050,3117,3118,3120,3123],{"class":1052,"line":2174},[1050,3119,1660],{"class":1162},[1050,3121,3122],{"class":1166},"Extract relevant response data for logging",[1050,3124,1170],{"class":1162},[1050,3126,3127,3129],{"class":1052,"line":2180},[1050,3128,1478],{"class":1063},[1050,3130,2773],{"class":1088},[1050,3132,3133,3135,3137,3139,3141,3143,3145,3147,3149,3151],{"class":1052,"line":2201},[1050,3134,2477],{"class":1260},[1050,3136,2780],{"class":1264},[1050,3138,1436],{"class":1260},[1050,3140,2785],{"class":1088},[1050,3142,2788],{"class":1067},[1050,3144,1089],{"class":1088},[1050,3146,2793],{"class":1126},[1050,3148,2796],{"class":1088},[1050,3150,2799],{"class":1126},[1050,3152,2802],{"class":1088},[1050,3154,3155,3157,3160,3162,3164,3166,3168,3170],{"class":1052,"line":2228},[1050,3156,2477],{"class":1260},[1050,3158,3159],{"class":1264},"status_code",[1050,3161,1436],{"class":1260},[1050,3163,2785],{"class":1088},[1050,3165,1631],{"class":1067},[1050,3167,1089],{"class":1088},[1050,3169,3159],{"class":1231},[1050,3171,2485],{"class":1088},[1050,3173,3174,3176,3178,3180,3182,3184,3186,3188,3190,3192,3195,3197,3199,3201],{"class":1052,"line":2259},[1050,3175,2477],{"class":1260},[1050,3177,2990],{"class":1264},[1050,3179,1436],{"class":1260},[1050,3181,2785],{"class":1088},[1050,3183,1631],{"class":1067},[1050,3185,1089],{"class":1088},[1050,3187,2967],{"class":1126},[1050,3189,1130],{"class":1088},[1050,3191,1436],{"class":1260},[1050,3193,3194],{"class":1264},"Content-Type",[1050,3196,1436],{"class":1260},[1050,3198,1195],{"class":1088},[1050,3200,2981],{"class":1260},[1050,3202,2866],{"class":1088},[1050,3204,3205,3207,3209,3211,3213,3216,3218,3220,3222,3225,3227,3229,3231,3233,3235,3237,3239,3241,3243,3245,3247,3249],{"class":1052,"line":2264},[1050,3206,2477],{"class":1260},[1050,3208,3026],{"class":1264},[1050,3210,1436],{"class":1260},[1050,3212,2785],{"class":1088},[1050,3214,3215],{"class":1185}," len",[1050,3217,1130],{"class":1088},[1050,3219,2013],{"class":1126},[1050,3221,1089],{"class":1088},[1050,3223,3224],{"class":1231},"content",[1050,3226,1931],{"class":1088},[1050,3228,2892],{"class":1063},[1050,3230,2008],{"class":1185},[1050,3232,1130],{"class":1088},[1050,3234,2013],{"class":1126},[1050,3236,1195],{"class":1088},[1050,3238,1915],{"class":1260},[1050,3240,3224],{"class":1264},[1050,3242,1436],{"class":1260},[1050,3244,1931],{"class":1088},[1050,3246,2911],{"class":1063},[1050,3248,3054],{"class":2331},[1050,3250,2485],{"class":1088},[1050,3252,3253,3255,3257,3259,3261,3264,3266,3268,3270,3273],{"class":1052,"line":2271},[1050,3254,2477],{"class":1260},[1050,3256,1458],{"class":1264},[1050,3258,1436],{"class":1260},[1050,3260,2785],{"class":1088},[1050,3262,3263],{"class":1185}," round",[1050,3265,1130],{"class":1088},[1050,3267,1458],{"class":1126},[1050,3269,1195],{"class":1088},[1050,3271,3272],{"class":2331}," 4",[1050,3274,2866],{"class":1088},[1050,3276,3277,3279,3281,3283,3285,3287,3289],{"class":1052,"line":2276},[1050,3278,2477],{"class":1260},[1050,3280,1752],{"class":1264},[1050,3282,1436],{"class":1260},[1050,3284,2785],{"class":1088},[1050,3286,1291],{"class":1067},[1050,3288,1089],{"class":1088},[1050,3290,3291],{"class":1231},"path\n",[1050,3293,3294],{"class":1052,"line":2290},[1050,3295,3085],{"class":1088},[1050,3297,3298],{"class":1052,"line":2300},[1050,3299,1176],{"class":1067},[1050,3301,3302,3304,3307,3309,3311,3313,3315],{"class":1052,"line":2309},[1050,3303,1182],{"class":1148},[1050,3305,3306],{"class":1644}," get_client_ip",[1050,3308,1130],{"class":1088},[1050,3310,1192],{"class":1191},[1050,3312,1195],{"class":1088},[1050,3314,1291],{"class":1198},[1050,3316,1202],{"class":1088},[1050,3318,3320,3322,3325],{"class":1052,"line":3319},61,[1050,3321,1660],{"class":1162},[1050,3323,3324],{"class":1166},"Get the client's IP address",[1050,3326,1170],{"class":1162},[1050,3328,3330,3333,3335,3337,3339,3341,3343,3345,3347,3349,3352,3354],{"class":1052,"line":3329},62,[1050,3331,3332],{"class":1067},"        x_forwarded_for ",[1050,3334,1118],{"class":1088},[1050,3336,1291],{"class":1067},[1050,3338,1089],{"class":1088},[1050,3340,2962],{"class":2862},[1050,3342,1089],{"class":1088},[1050,3344,2967],{"class":1126},[1050,3346,1130],{"class":1088},[1050,3348,1436],{"class":1260},[1050,3350,3351],{"class":1264},"HTTP_X_FORWARDED_FOR",[1050,3353,1436],{"class":1260},[1050,3355,1137],{"class":1088},[1050,3357,3359,3361,3364],{"class":1052,"line":3358},63,[1050,3360,2005],{"class":1063},[1050,3362,3363],{"class":1067}," x_forwarded_for",[1050,3365,1156],{"class":1088},[1050,3367,3369,3372,3374,3376,3378,3381,3383,3385,3387,3389,3392,3395],{"class":1052,"line":3368},64,[1050,3370,3371],{"class":1067},"            ip ",[1050,3373,1118],{"class":1088},[1050,3375,3363],{"class":1067},[1050,3377,1089],{"class":1088},[1050,3379,3380],{"class":1126},"split",[1050,3382,1130],{"class":1088},[1050,3384,1436],{"class":1260},[1050,3386,1195],{"class":1264},[1050,3388,1436],{"class":1260},[1050,3390,3391],{"class":1088},")[",[1050,3393,3394],{"class":2331},"0",[1050,3396,2335],{"class":1088},[1050,3398,3400,3403],{"class":1052,"line":3399},65,[1050,3401,3402],{"class":1063},"        else",[1050,3404,1156],{"class":1088},[1050,3406,3408,3410,3412,3414,3416,3418,3420,3422,3424,3426,3429,3431],{"class":1052,"line":3407},66,[1050,3409,3371],{"class":1067},[1050,3411,1118],{"class":1088},[1050,3413,1291],{"class":1067},[1050,3415,1089],{"class":1088},[1050,3417,2962],{"class":2862},[1050,3419,1089],{"class":1088},[1050,3421,2967],{"class":1126},[1050,3423,1130],{"class":1088},[1050,3425,1436],{"class":1260},[1050,3427,3428],{"class":1264},"REMOTE_ADDR",[1050,3430,1436],{"class":1260},[1050,3432,1137],{"class":1088},[1050,3434,3436,3438],{"class":1052,"line":3435},67,[1050,3437,1478],{"class":1063},[1050,3439,3440],{"class":1067}," ip\n",[1050,3442,3444],{"class":1052,"line":3443},68,[1050,3445,1176],{"class":1067},[1050,3447,3449,3451,3454,3456,3458,3460,3462],{"class":1052,"line":3448},69,[1050,3450,1182],{"class":1148},[1050,3452,3453],{"class":1644}," get_safe_headers",[1050,3455,1130],{"class":1088},[1050,3457,1192],{"class":1191},[1050,3459,1195],{"class":1088},[1050,3461,1291],{"class":1198},[1050,3463,1202],{"class":1088},[1050,3465,3467,3469,3472],{"class":1052,"line":3466},70,[1050,3468,1660],{"class":1162},[1050,3470,3471],{"class":1166},"Get headers excluding sensitive information",[1050,3473,1170],{"class":1162},[1050,3475,3477,3480,3482],{"class":1052,"line":3476},71,[1050,3478,3479],{"class":1067},"        headers ",[1050,3481,1118],{"class":1088},[1050,3483,3484],{"class":1088}," {}\n",[1050,3486,3488,3491,3494,3496,3499,3502,3504,3506,3508,3510,3513],{"class":1052,"line":3487},72,[1050,3489,3490],{"class":1063},"        for",[1050,3492,3493],{"class":1067}," key",[1050,3495,1195],{"class":1088},[1050,3497,3498],{"class":1067}," value ",[1050,3500,3501],{"class":1063},"in",[1050,3503,1291],{"class":1067},[1050,3505,1089],{"class":1088},[1050,3507,2962],{"class":2862},[1050,3509,1089],{"class":1088},[1050,3511,3512],{"class":1126},"items",[1050,3514,3515],{"class":1088},"():\n",[1050,3517,3519,3522,3524,3526,3529,3531,3533,3536,3538,3540,3543,3546,3549,3552,3554,3556,3558],{"class":1052,"line":3518},73,[1050,3520,3521],{"class":1063},"            if",[1050,3523,3493],{"class":1067},[1050,3525,1089],{"class":1088},[1050,3527,3528],{"class":1126},"startswith",[1050,3530,1130],{"class":1088},[1050,3532,1436],{"class":1260},[1050,3534,3535],{"class":1264},"HTTP_",[1050,3537,1436],{"class":1260},[1050,3539,1931],{"class":1088},[1050,3541,3542],{"class":1409}," and",[1050,3544,3545],{"class":1067}," key ",[1050,3547,3548],{"class":1409},"not",[1050,3550,3551],{"class":1409}," in",[1050,3553,1357],{"class":1225},[1050,3555,1089],{"class":1088},[1050,3557,2467],{"class":1231},[1050,3559,1156],{"class":1088},[1050,3561,3563,3566,3568,3570,3572,3575,3578,3581,3583,3585,3588,3590,3592,3594,3596,3598,3601,3604],{"class":1052,"line":3562},74,[1050,3564,3565],{"class":1067},"                header_name ",[1050,3567,1118],{"class":1088},[1050,3569,3493],{"class":1067},[1050,3571,1433],{"class":1088},[1050,3573,3574],{"class":2331},"5",[1050,3576,3577],{"class":1088},":].",[1050,3579,3580],{"class":1126},"replace",[1050,3582,1130],{"class":1088},[1050,3584,1436],{"class":1260},[1050,3586,3587],{"class":1264},"_",[1050,3589,1436],{"class":1260},[1050,3591,1195],{"class":1088},[1050,3593,1915],{"class":1260},[1050,3595,1410],{"class":1264},[1050,3597,1436],{"class":1260},[1050,3599,3600],{"class":1088},").",[1050,3602,3603],{"class":1126},"title",[1050,3605,1334],{"class":1088},[1050,3607,3609,3612,3614,3617,3619,3621],{"class":1052,"line":3608},75,[1050,3610,3611],{"class":1067},"                headers",[1050,3613,1433],{"class":1088},[1050,3615,3616],{"class":1067},"header_name",[1050,3618,1444],{"class":1088},[1050,3620,1235],{"class":1088},[1050,3622,3623],{"class":1067}," value\n",[1050,3625,3627,3629],{"class":1052,"line":3626},76,[1050,3628,1478],{"class":1063},[1050,3630,3631],{"class":1067}," headers\n",[1035,3633,3635],{"id":3634},"api-rate-limiting-middleware","API Rate Limiting Middleware",[1040,3637,3639],{"className":1042,"code":3638,"language":1044,"meta":1045,"style":1045},"from django.core.cache import cache\nfrom django.http import HttpResponseTooManyRequests\nimport json\n\nclass RateLimitMiddleware:\n    \"\"\"Rate limiting middleware for API endpoints\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n        self.rate_limits = {\n            '/api/': {'requests': 100, 'window': 3600},  # 100 requests per hour\n            '/api/auth/': {'requests': 10, 'window': 300},  # 10 requests per 5 minutes\n        }\n    \n    def __call__(self, request):\n        # Check if request should be rate limited\n        if self.should_rate_limit(request):\n            if self.is_rate_limited(request):\n                return self.rate_limit_response(request)\n        \n        response = self.get_response(request)\n        \n        # Add rate limit headers\n        if self.should_rate_limit(request):\n            self.add_rate_limit_headers(request, response)\n        \n        return response\n    \n    def should_rate_limit(self, request):\n        \"\"\"Determine if request should be rate limited\"\"\"\n        return any(request.path.startswith(path) for path in self.rate_limits.keys())\n    \n    def get_rate_limit_config(self, request):\n        \"\"\"Get rate limit configuration for request\"\"\"\n        for path, config in self.rate_limits.items():\n            if request.path.startswith(path):\n                return config\n        return None\n    \n    def is_rate_limited(self, request):\n        \"\"\"Check if request exceeds rate limit\"\"\"\n        config = self.get_rate_limit_config(request)\n        if not config:\n            return False\n        \n        # Create cache key based on IP and path\n        client_ip = self.get_client_ip(request)\n        cache_key = f\"rate_limit:{client_ip}:{request.path.split('/')[1]}\"\n        \n        # Get current request count\n        current_requests = cache.get(cache_key, 0)\n        \n        if current_requests >= config['requests']:\n            return True\n        \n        # Increment counter\n        cache.set(cache_key, current_requests + 1, config['window'])\n        return False\n    \n    def rate_limit_response(self, request):\n        \"\"\"Return rate limit exceeded response\"\"\"\n        config = self.get_rate_limit_config(request)\n        \n        response_data = {\n            'error': 'Rate limit exceeded',\n            'message': f\"Maximum {config['requests']} requests per {config['window']} seconds\",\n            'retry_after': config['window']\n        }\n        \n        response = HttpResponseTooManyRequests(\n            json.dumps(response_data),\n            content_type='application/json'\n        )\n        response['Retry-After'] = str(config['window'])\n        \n        return response\n    \n    def add_rate_limit_headers(self, request, response):\n        \"\"\"Add rate limit information to response headers\"\"\"\n        config = self.get_rate_limit_config(request)\n        if not config:\n            return\n        \n        client_ip = self.get_client_ip(request)\n        cache_key = f\"rate_limit:{client_ip}:{request.path.split('/')[1]}\"\n        current_requests = cache.get(cache_key, 0)\n        \n        response['X-RateLimit-Limit'] = str(config['requests'])\n        response['X-RateLimit-Remaining'] = str(max(0, config['requests'] - current_requests))\n        response['X-RateLimit-Reset'] = str(config['window'])\n    \n    def get_client_ip(self, request):\n        \"\"\"Get client IP address\"\"\"\n        x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')\n        if x_forwarded_for:\n            ip = x_forwarded_for.split(',')[0]\n        else:\n            ip = request.META.get('REMOTE_ADDR')\n        return ip\n",[1047,3640,3641,3662,3678,3684,3688,3697,3706,3710,3726,3738,3751,3797,3839,3843,3847,3863,3868,3885,3902,3920,3924,3942,3946,3951,3967,3987,3991,3997,4001,4018,4027,4074,4078,4095,4104,4130,4150,4157,4163,4167,4184,4193,4213,4225,4233,4237,4242,4261,4314,4318,4323,4348,4352,4375,4382,4386,4391,4430,4436,4440,4457,4466,4484,4488,4496,4515,4572,4595,4599,4603,4615,4630,4645,4650,4683,4687,4693,4698,4720,4730,4749,4760,4766,4771,4790,4839,4862,4867,4901,4952,4986,4991,5008,5018,5045,5054,5081,5088,5115],{"__ignoreMap":1045},[1050,3642,3643,3645,3647,3649,3652,3654,3657,3659],{"class":1052,"line":1053},[1050,3644,1082],{"class":1063},[1050,3646,1085],{"class":1067},[1050,3648,1089],{"class":1088},[1050,3650,3651],{"class":1067},"core",[1050,3653,1089],{"class":1088},[1050,3655,3656],{"class":1067},"cache ",[1050,3658,1064],{"class":1063},[1050,3660,3661],{"class":1067}," cache\n",[1050,3663,3664,3666,3668,3670,3673,3675],{"class":1052,"line":1060},[1050,3665,1082],{"class":1063},[1050,3667,1085],{"class":1067},[1050,3669,1089],{"class":1088},[1050,3671,3672],{"class":1067},"http ",[1050,3674,1064],{"class":1063},[1050,3676,3677],{"class":1067}," HttpResponseTooManyRequests\n",[1050,3679,3680,3682],{"class":1052,"line":1071},[1050,3681,1064],{"class":1063},[1050,3683,2355],{"class":1067},[1050,3685,3686],{"class":1052,"line":1079},[1050,3687,1109],{"emptyLinePlaceholder":1108},[1050,3689,3690,3692,3695],{"class":1052,"line":1105},[1050,3691,1149],{"class":1148},[1050,3693,3694],{"class":1152}," RateLimitMiddleware",[1050,3696,1156],{"class":1088},[1050,3698,3699,3701,3704],{"class":1052,"line":1112},[1050,3700,1163],{"class":1162},[1050,3702,3703],{"class":1166},"Rate limiting middleware for API endpoints",[1050,3705,1170],{"class":1162},[1050,3707,3708],{"class":1052,"line":1140},[1050,3709,1176],{"class":1067},[1050,3711,3712,3714,3716,3718,3720,3722,3724],{"class":1052,"line":1145},[1050,3713,1182],{"class":1148},[1050,3715,1186],{"class":1185},[1050,3717,1130],{"class":1088},[1050,3719,1192],{"class":1191},[1050,3721,1195],{"class":1088},[1050,3723,1199],{"class":1198},[1050,3725,1202],{"class":1088},[1050,3727,3728,3730,3732,3734,3736],{"class":1052,"line":1159},[1050,3729,1226],{"class":1225},[1050,3731,1089],{"class":1088},[1050,3733,1232],{"class":1231},[1050,3735,1235],{"class":1088},[1050,3737,1238],{"class":1067},[1050,3739,3740,3742,3744,3747,3749],{"class":1052,"line":1173},[1050,3741,1226],{"class":1225},[1050,3743,1089],{"class":1088},[1050,3745,3746],{"class":1231},"rate_limits",[1050,3748,1235],{"class":1088},[1050,3750,2773],{"class":1088},[1050,3752,3753,3755,3758,3760,3762,3765,3767,3770,3772,3774,3777,3779,3781,3784,3786,3788,3791,3794],{"class":1052,"line":1179},[1050,3754,2477],{"class":1260},[1050,3756,3757],{"class":1264},"/api/",[1050,3759,1436],{"class":1260},[1050,3761,2785],{"class":1088},[1050,3763,3764],{"class":1088}," {",[1050,3766,1436],{"class":1260},[1050,3768,3769],{"class":1264},"requests",[1050,3771,1436],{"class":1260},[1050,3773,2785],{"class":1088},[1050,3775,3776],{"class":2331}," 100",[1050,3778,1195],{"class":1088},[1050,3780,1915],{"class":1260},[1050,3782,3783],{"class":1264},"window",[1050,3785,1436],{"class":1260},[1050,3787,2785],{"class":1088},[1050,3789,3790],{"class":2331}," 3600",[1050,3792,3793],{"class":1088},"},",[1050,3795,3796],{"class":1056},"  # 100 requests per hour\n",[1050,3798,3799,3801,3804,3806,3808,3810,3812,3814,3816,3818,3821,3823,3825,3827,3829,3831,3834,3836],{"class":1052,"line":1205},[1050,3800,2477],{"class":1260},[1050,3802,3803],{"class":1264},"/api/auth/",[1050,3805,1436],{"class":1260},[1050,3807,2785],{"class":1088},[1050,3809,3764],{"class":1088},[1050,3811,1436],{"class":1260},[1050,3813,3769],{"class":1264},[1050,3815,1436],{"class":1260},[1050,3817,2785],{"class":1088},[1050,3819,3820],{"class":2331}," 10",[1050,3822,1195],{"class":1088},[1050,3824,1915],{"class":1260},[1050,3826,3783],{"class":1264},[1050,3828,1436],{"class":1260},[1050,3830,2785],{"class":1088},[1050,3832,3833],{"class":2331}," 300",[1050,3835,3793],{"class":1088},[1050,3837,3838],{"class":1056},"  # 10 requests per 5 minutes\n",[1050,3840,3841],{"class":1052,"line":1211},[1050,3842,3085],{"class":1088},[1050,3844,3845],{"class":1052,"line":1217},[1050,3846,1176],{"class":1067},[1050,3848,3849,3851,3853,3855,3857,3859,3861],{"class":1052,"line":1222},[1050,3850,1182],{"class":1148},[1050,3852,1282],{"class":1185},[1050,3854,1130],{"class":1088},[1050,3856,1192],{"class":1191},[1050,3858,1195],{"class":1088},[1050,3860,1291],{"class":1198},[1050,3862,1202],{"class":1088},[1050,3864,3865],{"class":1052,"line":1241},[1050,3866,3867],{"class":1056},"        # Check if request should be rate limited\n",[1050,3869,3870,3872,3874,3876,3879,3881,3883],{"class":1052,"line":1247},[1050,3871,2005],{"class":1063},[1050,3873,1357],{"class":1225},[1050,3875,1089],{"class":1088},[1050,3877,3878],{"class":1126},"should_rate_limit",[1050,3880,1130],{"class":1088},[1050,3882,1366],{"class":1126},[1050,3884,1202],{"class":1088},[1050,3886,3887,3889,3891,3893,3896,3898,3900],{"class":1052,"line":1272},[1050,3888,3521],{"class":1063},[1050,3890,1357],{"class":1225},[1050,3892,1089],{"class":1088},[1050,3894,3895],{"class":1126},"is_rate_limited",[1050,3897,1130],{"class":1088},[1050,3899,1366],{"class":1126},[1050,3901,1202],{"class":1088},[1050,3903,3904,3907,3909,3911,3914,3916,3918],{"class":1052,"line":1277},[1050,3905,3906],{"class":1063},"                return",[1050,3908,1357],{"class":1225},[1050,3910,1089],{"class":1088},[1050,3912,3913],{"class":1126},"rate_limit_response",[1050,3915,1130],{"class":1088},[1050,3917,1366],{"class":1126},[1050,3919,1137],{"class":1088},[1050,3921,3922],{"class":1052,"line":1296},[1050,3923,1340],{"class":1067},[1050,3925,3926,3928,3930,3932,3934,3936,3938,3940],{"class":1052,"line":1301},[1050,3927,1352],{"class":1067},[1050,3929,1118],{"class":1088},[1050,3931,1357],{"class":1225},[1050,3933,1089],{"class":1088},[1050,3935,1232],{"class":1126},[1050,3937,1130],{"class":1088},[1050,3939,1366],{"class":1126},[1050,3941,1137],{"class":1088},[1050,3943,3944],{"class":1052,"line":1307},[1050,3945,1340],{"class":1067},[1050,3947,3948],{"class":1052,"line":1312},[1050,3949,3950],{"class":1056},"        # Add rate limit headers\n",[1050,3952,3953,3955,3957,3959,3961,3963,3965],{"class":1052,"line":1318},[1050,3954,2005],{"class":1063},[1050,3956,1357],{"class":1225},[1050,3958,1089],{"class":1088},[1050,3960,3878],{"class":1126},[1050,3962,1130],{"class":1088},[1050,3964,1366],{"class":1126},[1050,3966,1202],{"class":1088},[1050,3968,3969,3972,3974,3977,3979,3981,3983,3985],{"class":1052,"line":1337},[1050,3970,3971],{"class":1225},"            self",[1050,3973,1089],{"class":1088},[1050,3975,3976],{"class":1126},"add_rate_limit_headers",[1050,3978,1130],{"class":1088},[1050,3980,1366],{"class":1126},[1050,3982,1195],{"class":1088},[1050,3984,1631],{"class":1126},[1050,3986,1137],{"class":1088},[1050,3988,3989],{"class":1052,"line":1343},[1050,3990,1340],{"class":1067},[1050,3992,3993,3995],{"class":1052,"line":1349},[1050,3994,1478],{"class":1063},[1050,3996,1481],{"class":1067},[1050,3998,3999],{"class":1052,"line":1371},[1050,4000,1176],{"class":1067},[1050,4002,4003,4005,4008,4010,4012,4014,4016],{"class":1052,"line":1376},[1050,4004,1182],{"class":1148},[1050,4006,4007],{"class":1644}," should_rate_limit",[1050,4009,1130],{"class":1088},[1050,4011,1192],{"class":1191},[1050,4013,1195],{"class":1088},[1050,4015,1291],{"class":1198},[1050,4017,1202],{"class":1088},[1050,4019,4020,4022,4025],{"class":1052,"line":1382},[1050,4021,1660],{"class":1162},[1050,4023,4024],{"class":1166},"Determine if request should be rate limited",[1050,4026,1170],{"class":1162},[1050,4028,4029,4031,4034,4036,4038,4040,4042,4044,4046,4048,4050,4052,4055,4058,4060,4062,4064,4066,4068,4071],{"class":1052,"line":1398},[1050,4030,1478],{"class":1063},[1050,4032,4033],{"class":1185}," any",[1050,4035,1130],{"class":1088},[1050,4037,1366],{"class":1126},[1050,4039,1089],{"class":1088},[1050,4041,1752],{"class":1231},[1050,4043,1089],{"class":1088},[1050,4045,3528],{"class":1126},[1050,4047,1130],{"class":1088},[1050,4049,1752],{"class":1126},[1050,4051,1931],{"class":1088},[1050,4053,4054],{"class":1063}," for",[1050,4056,4057],{"class":1126}," path ",[1050,4059,3501],{"class":1063},[1050,4061,1357],{"class":1225},[1050,4063,1089],{"class":1088},[1050,4065,3746],{"class":1231},[1050,4067,1089],{"class":1088},[1050,4069,4070],{"class":1126},"keys",[1050,4072,4073],{"class":1088},"())\n",[1050,4075,4076],{"class":1052,"line":1416},[1050,4077,1176],{"class":1067},[1050,4079,4080,4082,4085,4087,4089,4091,4093],{"class":1052,"line":1421},[1050,4081,1182],{"class":1148},[1050,4083,4084],{"class":1644}," get_rate_limit_config",[1050,4086,1130],{"class":1088},[1050,4088,1192],{"class":1191},[1050,4090,1195],{"class":1088},[1050,4092,1291],{"class":1198},[1050,4094,1202],{"class":1088},[1050,4096,4097,4099,4102],{"class":1052,"line":1427},[1050,4098,1660],{"class":1162},[1050,4100,4101],{"class":1166},"Get rate limit configuration for request",[1050,4103,1170],{"class":1162},[1050,4105,4106,4108,4111,4113,4116,4118,4120,4122,4124,4126,4128],{"class":1052,"line":1470},[1050,4107,3490],{"class":1063},[1050,4109,4110],{"class":1067}," path",[1050,4112,1195],{"class":1088},[1050,4114,4115],{"class":1067}," config ",[1050,4117,3501],{"class":1063},[1050,4119,1357],{"class":1225},[1050,4121,1089],{"class":1088},[1050,4123,3746],{"class":1231},[1050,4125,1089],{"class":1088},[1050,4127,3512],{"class":1126},[1050,4129,3515],{"class":1088},[1050,4131,4132,4134,4136,4138,4140,4142,4144,4146,4148],{"class":1052,"line":1475},[1050,4133,3521],{"class":1063},[1050,4135,1291],{"class":1067},[1050,4137,1089],{"class":1088},[1050,4139,1752],{"class":1231},[1050,4141,1089],{"class":1088},[1050,4143,3528],{"class":1126},[1050,4145,1130],{"class":1088},[1050,4147,1752],{"class":1126},[1050,4149,1202],{"class":1088},[1050,4151,4152,4154],{"class":1052,"line":1965},[1050,4153,3906],{"class":1063},[1050,4155,4156],{"class":1067}," config\n",[1050,4158,4159,4161],{"class":1052,"line":1970},[1050,4160,1478],{"class":1063},[1050,4162,1850],{"class":1849},[1050,4164,4165],{"class":1052,"line":1992},[1050,4166,1176],{"class":1067},[1050,4168,4169,4171,4174,4176,4178,4180,4182],{"class":1052,"line":2002},[1050,4170,1182],{"class":1148},[1050,4172,4173],{"class":1644}," is_rate_limited",[1050,4175,1130],{"class":1088},[1050,4177,1192],{"class":1191},[1050,4179,1195],{"class":1088},[1050,4181,1291],{"class":1198},[1050,4183,1202],{"class":1088},[1050,4185,4186,4188,4191],{"class":1052,"line":2027},[1050,4187,1660],{"class":1162},[1050,4189,4190],{"class":1166},"Check if request exceeds rate limit",[1050,4192,1170],{"class":1162},[1050,4194,4195,4198,4200,4202,4204,4207,4209,4211],{"class":1052,"line":2072},[1050,4196,4197],{"class":1067},"        config ",[1050,4199,1118],{"class":1088},[1050,4201,1357],{"class":1225},[1050,4203,1089],{"class":1088},[1050,4205,4206],{"class":1126},"get_rate_limit_config",[1050,4208,1130],{"class":1088},[1050,4210,1366],{"class":1126},[1050,4212,1137],{"class":1088},[1050,4214,4215,4217,4220,4223],{"class":1052,"line":2077},[1050,4216,2005],{"class":1063},[1050,4218,4219],{"class":1409}," not",[1050,4221,4222],{"class":1067}," config",[1050,4224,1156],{"class":1088},[1050,4226,4227,4230],{"class":1052,"line":2084},[1050,4228,4229],{"class":1063},"            return",[1050,4231,4232],{"class":1849}," False\n",[1050,4234,4235],{"class":1052,"line":2089},[1050,4236,1340],{"class":1067},[1050,4238,4239],{"class":1052,"line":2111},[1050,4240,4241],{"class":1056},"        # Create cache key based on IP and path\n",[1050,4243,4244,4247,4249,4251,4253,4255,4257,4259],{"class":1052,"line":2121},[1050,4245,4246],{"class":1067},"        client_ip ",[1050,4248,1118],{"class":1088},[1050,4250,1357],{"class":1225},[1050,4252,1089],{"class":1088},[1050,4254,2938],{"class":1126},[1050,4256,1130],{"class":1088},[1050,4258,1366],{"class":1126},[1050,4260,1137],{"class":1088},[1050,4262,4263,4266,4268,4270,4273,4275,4278,4280,4282,4284,4286,4288,4290,4292,4294,4296,4298,4301,4303,4305,4308,4310,4312],{"class":1052,"line":2127},[1050,4264,4265],{"class":1067},"        cache_key ",[1050,4267,1118],{"class":1088},[1050,4269,1449],{"class":1148},[1050,4271,4272],{"class":1264},"\"rate_limit:",[1050,4274,1455],{"class":1454},[1050,4276,4277],{"class":1067},"client_ip",[1050,4279,1464],{"class":1454},[1050,4281,2785],{"class":1264},[1050,4283,1455],{"class":1454},[1050,4285,1366],{"class":1067},[1050,4287,1089],{"class":1088},[1050,4289,1752],{"class":1231},[1050,4291,1089],{"class":1088},[1050,4293,3380],{"class":1126},[1050,4295,1130],{"class":1088},[1050,4297,1436],{"class":1260},[1050,4299,4300],{"class":1264},"/",[1050,4302,1436],{"class":1260},[1050,4304,3391],{"class":1088},[1050,4306,4307],{"class":2331},"1",[1050,4309,1444],{"class":1088},[1050,4311,1464],{"class":1454},[1050,4313,1467],{"class":1264},[1050,4315,4316],{"class":1052,"line":2169},[1050,4317,1340],{"class":1067},[1050,4319,4320],{"class":1052,"line":2174},[1050,4321,4322],{"class":1056},"        # Get current request count\n",[1050,4324,4325,4328,4330,4333,4335,4337,4339,4342,4344,4346],{"class":1052,"line":2180},[1050,4326,4327],{"class":1067},"        current_requests ",[1050,4329,1118],{"class":1088},[1050,4331,4332],{"class":1067}," cache",[1050,4334,1089],{"class":1088},[1050,4336,2967],{"class":1126},[1050,4338,1130],{"class":1088},[1050,4340,4341],{"class":1126},"cache_key",[1050,4343,1195],{"class":1088},[1050,4345,3054],{"class":2331},[1050,4347,1137],{"class":1088},[1050,4349,4350],{"class":1052,"line":2201},[1050,4351,1340],{"class":1067},[1050,4353,4354,4356,4359,4362,4364,4366,4368,4370,4372],{"class":1052,"line":2228},[1050,4355,2005],{"class":1063},[1050,4357,4358],{"class":1067}," current_requests ",[1050,4360,4361],{"class":1409},">=",[1050,4363,4222],{"class":1067},[1050,4365,1433],{"class":1088},[1050,4367,1436],{"class":1260},[1050,4369,3769],{"class":1264},[1050,4371,1436],{"class":1260},[1050,4373,4374],{"class":1088},"]:\n",[1050,4376,4377,4379],{"class":1052,"line":2259},[1050,4378,4229],{"class":1063},[1050,4380,4381],{"class":1849}," True\n",[1050,4383,4384],{"class":1052,"line":2264},[1050,4385,1340],{"class":1067},[1050,4387,4388],{"class":1052,"line":2271},[1050,4389,4390],{"class":1056},"        # Increment counter\n",[1050,4392,4393,4396,4398,4401,4403,4405,4407,4409,4412,4415,4417,4419,4421,4423,4425,4427],{"class":1052,"line":2276},[1050,4394,4395],{"class":1067},"        cache",[1050,4397,1089],{"class":1088},[1050,4399,4400],{"class":1126},"set",[1050,4402,1130],{"class":1088},[1050,4404,4341],{"class":1126},[1050,4406,1195],{"class":1088},[1050,4408,4358],{"class":1126},[1050,4410,4411],{"class":1409},"+",[1050,4413,4414],{"class":2331}," 1",[1050,4416,1195],{"class":1088},[1050,4418,4222],{"class":1126},[1050,4420,1433],{"class":1088},[1050,4422,1436],{"class":1260},[1050,4424,3783],{"class":1264},[1050,4426,1436],{"class":1260},[1050,4428,4429],{"class":1088},"])\n",[1050,4431,4432,4434],{"class":1052,"line":2290},[1050,4433,1478],{"class":1063},[1050,4435,4232],{"class":1849},[1050,4437,4438],{"class":1052,"line":2300},[1050,4439,1176],{"class":1067},[1050,4441,4442,4444,4447,4449,4451,4453,4455],{"class":1052,"line":2309},[1050,4443,1182],{"class":1148},[1050,4445,4446],{"class":1644}," rate_limit_response",[1050,4448,1130],{"class":1088},[1050,4450,1192],{"class":1191},[1050,4452,1195],{"class":1088},[1050,4454,1291],{"class":1198},[1050,4456,1202],{"class":1088},[1050,4458,4459,4461,4464],{"class":1052,"line":3319},[1050,4460,1660],{"class":1162},[1050,4462,4463],{"class":1166},"Return rate limit exceeded response",[1050,4465,1170],{"class":1162},[1050,4467,4468,4470,4472,4474,4476,4478,4480,4482],{"class":1052,"line":3329},[1050,4469,4197],{"class":1067},[1050,4471,1118],{"class":1088},[1050,4473,1357],{"class":1225},[1050,4475,1089],{"class":1088},[1050,4477,4206],{"class":1126},[1050,4479,1130],{"class":1088},[1050,4481,1366],{"class":1126},[1050,4483,1137],{"class":1088},[1050,4485,4486],{"class":1052,"line":3358},[1050,4487,1340],{"class":1067},[1050,4489,4490,4492,4494],{"class":1052,"line":3368},[1050,4491,2662],{"class":1067},[1050,4493,1118],{"class":1088},[1050,4495,2773],{"class":1088},[1050,4497,4498,4500,4502,4504,4506,4508,4511,4513],{"class":1052,"line":3399},[1050,4499,2477],{"class":1260},[1050,4501,1894],{"class":1264},[1050,4503,1436],{"class":1260},[1050,4505,2785],{"class":1088},[1050,4507,1915],{"class":1260},[1050,4509,4510],{"class":1264},"Rate limit exceeded",[1050,4512,1436],{"class":1260},[1050,4514,2485],{"class":1088},[1050,4516,4517,4519,4522,4524,4526,4528,4531,4533,4536,4538,4540,4542,4544,4546,4548,4551,4553,4555,4557,4559,4561,4563,4565,4567,4570],{"class":1052,"line":3407},[1050,4518,2477],{"class":1260},[1050,4520,4521],{"class":1264},"message",[1050,4523,1436],{"class":1260},[1050,4525,2785],{"class":1088},[1050,4527,1449],{"class":1148},[1050,4529,4530],{"class":1264},"\"Maximum ",[1050,4532,1455],{"class":1454},[1050,4534,4535],{"class":1067},"config",[1050,4537,1433],{"class":1088},[1050,4539,1436],{"class":1260},[1050,4541,3769],{"class":1264},[1050,4543,1436],{"class":1260},[1050,4545,1444],{"class":1088},[1050,4547,1464],{"class":1454},[1050,4549,4550],{"class":1264}," requests per ",[1050,4552,1455],{"class":1454},[1050,4554,4535],{"class":1067},[1050,4556,1433],{"class":1088},[1050,4558,1436],{"class":1260},[1050,4560,3783],{"class":1264},[1050,4562,1436],{"class":1260},[1050,4564,1444],{"class":1088},[1050,4566,1464],{"class":1454},[1050,4568,4569],{"class":1264}," seconds\"",[1050,4571,2485],{"class":1088},[1050,4573,4574,4576,4579,4581,4583,4585,4587,4589,4591,4593],{"class":1052,"line":3435},[1050,4575,2477],{"class":1260},[1050,4577,4578],{"class":1264},"retry_after",[1050,4580,1436],{"class":1260},[1050,4582,2785],{"class":1088},[1050,4584,4222],{"class":1067},[1050,4586,1433],{"class":1088},[1050,4588,1436],{"class":1260},[1050,4590,3783],{"class":1264},[1050,4592,1436],{"class":1260},[1050,4594,2335],{"class":1088},[1050,4596,4597],{"class":1052,"line":3443},[1050,4598,3085],{"class":1088},[1050,4600,4601],{"class":1052,"line":3448},[1050,4602,1340],{"class":1067},[1050,4604,4605,4607,4609,4612],{"class":1052,"line":3466},[1050,4606,1352],{"class":1067},[1050,4608,1118],{"class":1088},[1050,4610,4611],{"class":1126}," HttpResponseTooManyRequests",[1050,4613,4614],{"class":1088},"(\n",[1050,4616,4617,4620,4622,4624,4626,4628],{"class":1052,"line":3476},[1050,4618,4619],{"class":1126},"            json",[1050,4621,1089],{"class":1088},[1050,4623,2581],{"class":1126},[1050,4625,1130],{"class":1088},[1050,4627,2718],{"class":1126},[1050,4629,2866],{"class":1088},[1050,4631,4632,4636,4638,4640,4643],{"class":1052,"line":3487},[1050,4633,4635],{"class":4634},"sqOPj","            content_type",[1050,4637,1118],{"class":1088},[1050,4639,1436],{"class":1260},[1050,4641,4642],{"class":1264},"application/json",[1050,4644,2506],{"class":1260},[1050,4646,4647],{"class":1052,"line":3518},[1050,4648,4649],{"class":1088},"        )\n",[1050,4651,4652,4654,4656,4658,4661,4663,4665,4667,4669,4671,4673,4675,4677,4679,4681],{"class":1052,"line":3562},[1050,4653,1430],{"class":1067},[1050,4655,1433],{"class":1088},[1050,4657,1436],{"class":1260},[1050,4659,4660],{"class":1264},"Retry-After",[1050,4662,1436],{"class":1260},[1050,4664,1444],{"class":1088},[1050,4666,1235],{"class":1088},[1050,4668,2315],{"class":2314},[1050,4670,1130],{"class":1088},[1050,4672,4535],{"class":1126},[1050,4674,1433],{"class":1088},[1050,4676,1436],{"class":1260},[1050,4678,3783],{"class":1264},[1050,4680,1436],{"class":1260},[1050,4682,4429],{"class":1088},[1050,4684,4685],{"class":1052,"line":3608},[1050,4686,1340],{"class":1067},[1050,4688,4689,4691],{"class":1052,"line":3626},[1050,4690,1478],{"class":1063},[1050,4692,1481],{"class":1067},[1050,4694,4696],{"class":1052,"line":4695},77,[1050,4697,1176],{"class":1067},[1050,4699,4701,4703,4706,4708,4710,4712,4714,4716,4718],{"class":1052,"line":4700},78,[1050,4702,1182],{"class":1148},[1050,4704,4705],{"class":1644}," add_rate_limit_headers",[1050,4707,1130],{"class":1088},[1050,4709,1192],{"class":1191},[1050,4711,1195],{"class":1088},[1050,4713,1291],{"class":1198},[1050,4715,1195],{"class":1088},[1050,4717,1631],{"class":1198},[1050,4719,1202],{"class":1088},[1050,4721,4723,4725,4728],{"class":1052,"line":4722},79,[1050,4724,1660],{"class":1162},[1050,4726,4727],{"class":1166},"Add rate limit information to response headers",[1050,4729,1170],{"class":1162},[1050,4731,4733,4735,4737,4739,4741,4743,4745,4747],{"class":1052,"line":4732},80,[1050,4734,4197],{"class":1067},[1050,4736,1118],{"class":1088},[1050,4738,1357],{"class":1225},[1050,4740,1089],{"class":1088},[1050,4742,4206],{"class":1126},[1050,4744,1130],{"class":1088},[1050,4746,1366],{"class":1126},[1050,4748,1137],{"class":1088},[1050,4750,4752,4754,4756,4758],{"class":1052,"line":4751},81,[1050,4753,2005],{"class":1063},[1050,4755,4219],{"class":1409},[1050,4757,4222],{"class":1067},[1050,4759,1156],{"class":1088},[1050,4761,4763],{"class":1052,"line":4762},82,[1050,4764,4765],{"class":1063},"            return\n",[1050,4767,4769],{"class":1052,"line":4768},83,[1050,4770,1340],{"class":1067},[1050,4772,4774,4776,4778,4780,4782,4784,4786,4788],{"class":1052,"line":4773},84,[1050,4775,4246],{"class":1067},[1050,4777,1118],{"class":1088},[1050,4779,1357],{"class":1225},[1050,4781,1089],{"class":1088},[1050,4783,2938],{"class":1126},[1050,4785,1130],{"class":1088},[1050,4787,1366],{"class":1126},[1050,4789,1137],{"class":1088},[1050,4791,4793,4795,4797,4799,4801,4803,4805,4807,4809,4811,4813,4815,4817,4819,4821,4823,4825,4827,4829,4831,4833,4835,4837],{"class":1052,"line":4792},85,[1050,4794,4265],{"class":1067},[1050,4796,1118],{"class":1088},[1050,4798,1449],{"class":1148},[1050,4800,4272],{"class":1264},[1050,4802,1455],{"class":1454},[1050,4804,4277],{"class":1067},[1050,4806,1464],{"class":1454},[1050,4808,2785],{"class":1264},[1050,4810,1455],{"class":1454},[1050,4812,1366],{"class":1067},[1050,4814,1089],{"class":1088},[1050,4816,1752],{"class":1231},[1050,4818,1089],{"class":1088},[1050,4820,3380],{"class":1126},[1050,4822,1130],{"class":1088},[1050,4824,1436],{"class":1260},[1050,4826,4300],{"class":1264},[1050,4828,1436],{"class":1260},[1050,4830,3391],{"class":1088},[1050,4832,4307],{"class":2331},[1050,4834,1444],{"class":1088},[1050,4836,1464],{"class":1454},[1050,4838,1467],{"class":1264},[1050,4840,4842,4844,4846,4848,4850,4852,4854,4856,4858,4860],{"class":1052,"line":4841},86,[1050,4843,4327],{"class":1067},[1050,4845,1118],{"class":1088},[1050,4847,4332],{"class":1067},[1050,4849,1089],{"class":1088},[1050,4851,2967],{"class":1126},[1050,4853,1130],{"class":1088},[1050,4855,4341],{"class":1126},[1050,4857,1195],{"class":1088},[1050,4859,3054],{"class":2331},[1050,4861,1137],{"class":1088},[1050,4863,4865],{"class":1052,"line":4864},87,[1050,4866,1340],{"class":1067},[1050,4868,4870,4872,4874,4876,4879,4881,4883,4885,4887,4889,4891,4893,4895,4897,4899],{"class":1052,"line":4869},88,[1050,4871,1430],{"class":1067},[1050,4873,1433],{"class":1088},[1050,4875,1436],{"class":1260},[1050,4877,4878],{"class":1264},"X-RateLimit-Limit",[1050,4880,1436],{"class":1260},[1050,4882,1444],{"class":1088},[1050,4884,1235],{"class":1088},[1050,4886,2315],{"class":2314},[1050,4888,1130],{"class":1088},[1050,4890,4535],{"class":1126},[1050,4892,1433],{"class":1088},[1050,4894,1436],{"class":1260},[1050,4896,3769],{"class":1264},[1050,4898,1436],{"class":1260},[1050,4900,4429],{"class":1088},[1050,4902,4904,4906,4908,4910,4913,4915,4917,4919,4921,4923,4926,4928,4930,4932,4934,4936,4938,4940,4942,4944,4946,4949],{"class":1052,"line":4903},89,[1050,4905,1430],{"class":1067},[1050,4907,1433],{"class":1088},[1050,4909,1436],{"class":1260},[1050,4911,4912],{"class":1264},"X-RateLimit-Remaining",[1050,4914,1436],{"class":1260},[1050,4916,1444],{"class":1088},[1050,4918,1235],{"class":1088},[1050,4920,2315],{"class":2314},[1050,4922,1130],{"class":1088},[1050,4924,4925],{"class":1185},"max",[1050,4927,1130],{"class":1088},[1050,4929,3394],{"class":2331},[1050,4931,1195],{"class":1088},[1050,4933,4222],{"class":1126},[1050,4935,1433],{"class":1088},[1050,4937,1436],{"class":1260},[1050,4939,3769],{"class":1264},[1050,4941,1436],{"class":1260},[1050,4943,1444],{"class":1088},[1050,4945,2218],{"class":1409},[1050,4947,4948],{"class":1126}," current_requests",[1050,4950,4951],{"class":1088},"))\n",[1050,4953,4955,4957,4959,4961,4964,4966,4968,4970,4972,4974,4976,4978,4980,4982,4984],{"class":1052,"line":4954},90,[1050,4956,1430],{"class":1067},[1050,4958,1433],{"class":1088},[1050,4960,1436],{"class":1260},[1050,4962,4963],{"class":1264},"X-RateLimit-Reset",[1050,4965,1436],{"class":1260},[1050,4967,1444],{"class":1088},[1050,4969,1235],{"class":1088},[1050,4971,2315],{"class":2314},[1050,4973,1130],{"class":1088},[1050,4975,4535],{"class":1126},[1050,4977,1433],{"class":1088},[1050,4979,1436],{"class":1260},[1050,4981,3783],{"class":1264},[1050,4983,1436],{"class":1260},[1050,4985,4429],{"class":1088},[1050,4987,4989],{"class":1052,"line":4988},91,[1050,4990,1176],{"class":1067},[1050,4992,4994,4996,4998,5000,5002,5004,5006],{"class":1052,"line":4993},92,[1050,4995,1182],{"class":1148},[1050,4997,3306],{"class":1644},[1050,4999,1130],{"class":1088},[1050,5001,1192],{"class":1191},[1050,5003,1195],{"class":1088},[1050,5005,1291],{"class":1198},[1050,5007,1202],{"class":1088},[1050,5009,5011,5013,5016],{"class":1052,"line":5010},93,[1050,5012,1660],{"class":1162},[1050,5014,5015],{"class":1166},"Get client IP address",[1050,5017,1170],{"class":1162},[1050,5019,5021,5023,5025,5027,5029,5031,5033,5035,5037,5039,5041,5043],{"class":1052,"line":5020},94,[1050,5022,3332],{"class":1067},[1050,5024,1118],{"class":1088},[1050,5026,1291],{"class":1067},[1050,5028,1089],{"class":1088},[1050,5030,2962],{"class":2862},[1050,5032,1089],{"class":1088},[1050,5034,2967],{"class":1126},[1050,5036,1130],{"class":1088},[1050,5038,1436],{"class":1260},[1050,5040,3351],{"class":1264},[1050,5042,1436],{"class":1260},[1050,5044,1137],{"class":1088},[1050,5046,5048,5050,5052],{"class":1052,"line":5047},95,[1050,5049,2005],{"class":1063},[1050,5051,3363],{"class":1067},[1050,5053,1156],{"class":1088},[1050,5055,5057,5059,5061,5063,5065,5067,5069,5071,5073,5075,5077,5079],{"class":1052,"line":5056},96,[1050,5058,3371],{"class":1067},[1050,5060,1118],{"class":1088},[1050,5062,3363],{"class":1067},[1050,5064,1089],{"class":1088},[1050,5066,3380],{"class":1126},[1050,5068,1130],{"class":1088},[1050,5070,1436],{"class":1260},[1050,5072,1195],{"class":1264},[1050,5074,1436],{"class":1260},[1050,5076,3391],{"class":1088},[1050,5078,3394],{"class":2331},[1050,5080,2335],{"class":1088},[1050,5082,5084,5086],{"class":1052,"line":5083},97,[1050,5085,3402],{"class":1063},[1050,5087,1156],{"class":1088},[1050,5089,5091,5093,5095,5097,5099,5101,5103,5105,5107,5109,5111,5113],{"class":1052,"line":5090},98,[1050,5092,3371],{"class":1067},[1050,5094,1118],{"class":1088},[1050,5096,1291],{"class":1067},[1050,5098,1089],{"class":1088},[1050,5100,2962],{"class":2862},[1050,5102,1089],{"class":1088},[1050,5104,2967],{"class":1126},[1050,5106,1130],{"class":1088},[1050,5108,1436],{"class":1260},[1050,5110,3428],{"class":1264},[1050,5112,1436],{"class":1260},[1050,5114,1137],{"class":1088},[1050,5116,5118,5120],{"class":1052,"line":5117},99,[1050,5119,1478],{"class":1063},[1050,5121,3440],{"class":1067},[1035,5123,5125],{"id":5124},"user-activity-tracking-middleware","User Activity Tracking Middleware",[1040,5127,5129],{"className":1042,"code":5128,"language":1044,"meta":1045,"style":1045},"from django.contrib.auth.models import AnonymousUser\nfrom django.utils import timezone\nfrom .models import UserActivity\n\nclass UserActivityMiddleware:\n    \"\"\"Track user activity and update last seen timestamp\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n        self.tracked_paths = ['/dashboard/', '/profile/', '/api/']\n        self.excluded_paths = ['/static/', '/media/', '/favicon.ico']\n    \n    def __call__(self, request):\n        response = self.get_response(request)\n        \n        # Track activity after successful response\n        if (response.status_code \u003C 400 and \n            self.should_track_activity(request)):\n            self.track_user_activity(request, response)\n        \n        return response\n    \n    def should_track_activity(self, request):\n        \"\"\"Determine if activity should be tracked\"\"\"\n        # Don't track static files\n        if any(request.path.startswith(path) for path in self.excluded_paths):\n            return False\n        \n        # Only track authenticated users\n        if not hasattr(request, 'user') or isinstance(request.user, AnonymousUser):\n            return False\n        \n        # Only track specific paths or all if no specific paths defined\n        if self.tracked_paths:\n            return any(request.path.startswith(path) for path in self.tracked_paths)\n        \n        return True\n    \n    def track_user_activity(self, request, response):\n        \"\"\"Record user activity\"\"\"\n        try:\n            activity_data = {\n                'user': request.user,\n                'path': request.path,\n                'method': request.method,\n                'ip_address': self.get_client_ip(request),\n                'user_agent': request.META.get('HTTP_USER_AGENT', ''),\n                'timestamp': timezone.now(),\n                'status_code': response.status_code\n            }\n            \n            # Create activity record\n            UserActivity.objects.create(**activity_data)\n            \n            # Update user's last activity timestamp\n            request.user.last_activity = timezone.now()\n            request.user.save(update_fields=['last_activity'])\n            \n        except Exception as e:\n            # Log error but don't break the request\n            logger.error(f\"Error tracking user activity: {e}\")\n    \n    def get_client_ip(self, request):\n        \"\"\"Get client IP address\"\"\"\n        x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')\n        if x_forwarded_for:\n            ip = x_forwarded_for.split(',')[0]\n        else:\n            ip = request.META.get('REMOTE_ADDR')\n        return ip\n\n# models.py\nclass UserActivity(models.Model):\n    \"\"\"Model to store user activity\"\"\"\n    user = models.ForeignKey(User, on_delete=models.CASCADE)\n    path = models.CharField(max_length=255)\n    method = models.CharField(max_length=10)\n    ip_address = models.GenericIPAddressField()\n    user_agent = models.TextField()\n    timestamp = models.DateTimeField()\n    status_code = models.IntegerField()\n    \n    class Meta:\n        ordering = ['-timestamp']\n        indexes = [\n            models.Index(fields=['user', 'timestamp']),\n            models.Index(fields=['timestamp']),\n        ]\n",[1047,5130,5131,5157,5171,5185,5189,5198,5207,5211,5227,5239,5279,5319,5323,5339,5357,5361,5366,5390,5406,5425,5429,5435,5439,5456,5465,5470,5508,5514,5518,5523,5566,5572,5576,5581,5593,5631,5635,5641,5645,5666,5675,5682,5691,5710,5728,5746,5768,5802,5820,5837,5842,5847,5852,5877,5881,5886,5910,5939,5943,5959,5964,5991,5995,6011,6019,6045,6053,6079,6085,6111,6117,6121,6126,6146,6155,6191,6217,6241,6257,6273,6289,6305,6309,6319,6337,6346,6380,6402],{"__ignoreMap":1045},[1050,5132,5133,5135,5137,5139,5142,5144,5147,5149,5152,5154],{"class":1052,"line":1053},[1050,5134,1082],{"class":1063},[1050,5136,1085],{"class":1067},[1050,5138,1089],{"class":1088},[1050,5140,5141],{"class":1067},"contrib",[1050,5143,1089],{"class":1088},[1050,5145,5146],{"class":1067},"auth",[1050,5148,1089],{"class":1088},[1050,5150,5151],{"class":1067},"models ",[1050,5153,1064],{"class":1063},[1050,5155,5156],{"class":1067}," AnonymousUser\n",[1050,5158,5159,5161,5163,5165,5167,5169],{"class":1052,"line":1060},[1050,5160,1082],{"class":1063},[1050,5162,1085],{"class":1067},[1050,5164,1089],{"class":1088},[1050,5166,2372],{"class":1067},[1050,5168,1064],{"class":1063},[1050,5170,2377],{"class":1067},[1050,5172,5173,5175,5178,5180,5182],{"class":1052,"line":1071},[1050,5174,1082],{"class":1063},[1050,5176,5177],{"class":1088}," .",[1050,5179,5151],{"class":1067},[1050,5181,1064],{"class":1063},[1050,5183,5184],{"class":1067}," UserActivity\n",[1050,5186,5187],{"class":1052,"line":1079},[1050,5188,1109],{"emptyLinePlaceholder":1108},[1050,5190,5191,5193,5196],{"class":1052,"line":1105},[1050,5192,1149],{"class":1148},[1050,5194,5195],{"class":1152}," UserActivityMiddleware",[1050,5197,1156],{"class":1088},[1050,5199,5200,5202,5205],{"class":1052,"line":1112},[1050,5201,1163],{"class":1162},[1050,5203,5204],{"class":1166},"Track user activity and update last seen timestamp",[1050,5206,1170],{"class":1162},[1050,5208,5209],{"class":1052,"line":1140},[1050,5210,1176],{"class":1067},[1050,5212,5213,5215,5217,5219,5221,5223,5225],{"class":1052,"line":1145},[1050,5214,1182],{"class":1148},[1050,5216,1186],{"class":1185},[1050,5218,1130],{"class":1088},[1050,5220,1192],{"class":1191},[1050,5222,1195],{"class":1088},[1050,5224,1199],{"class":1198},[1050,5226,1202],{"class":1088},[1050,5228,5229,5231,5233,5235,5237],{"class":1052,"line":1159},[1050,5230,1226],{"class":1225},[1050,5232,1089],{"class":1088},[1050,5234,1232],{"class":1231},[1050,5236,1235],{"class":1088},[1050,5238,1238],{"class":1067},[1050,5240,5241,5243,5245,5248,5250,5253,5255,5258,5260,5262,5264,5267,5269,5271,5273,5275,5277],{"class":1052,"line":1173},[1050,5242,1226],{"class":1225},[1050,5244,1089],{"class":1088},[1050,5246,5247],{"class":1231},"tracked_paths",[1050,5249,1235],{"class":1088},[1050,5251,5252],{"class":1088}," [",[1050,5254,1436],{"class":1260},[1050,5256,5257],{"class":1264},"/dashboard/",[1050,5259,1436],{"class":1260},[1050,5261,1195],{"class":1088},[1050,5263,1915],{"class":1260},[1050,5265,5266],{"class":1264},"/profile/",[1050,5268,1436],{"class":1260},[1050,5270,1195],{"class":1088},[1050,5272,1915],{"class":1260},[1050,5274,3757],{"class":1264},[1050,5276,1436],{"class":1260},[1050,5278,2335],{"class":1088},[1050,5280,5281,5283,5285,5288,5290,5292,5294,5297,5299,5301,5303,5306,5308,5310,5312,5315,5317],{"class":1052,"line":1179},[1050,5282,1226],{"class":1225},[1050,5284,1089],{"class":1088},[1050,5286,5287],{"class":1231},"excluded_paths",[1050,5289,1235],{"class":1088},[1050,5291,5252],{"class":1088},[1050,5293,1436],{"class":1260},[1050,5295,5296],{"class":1264},"/static/",[1050,5298,1436],{"class":1260},[1050,5300,1195],{"class":1088},[1050,5302,1915],{"class":1260},[1050,5304,5305],{"class":1264},"/media/",[1050,5307,1436],{"class":1260},[1050,5309,1195],{"class":1088},[1050,5311,1915],{"class":1260},[1050,5313,5314],{"class":1264},"/favicon.ico",[1050,5316,1436],{"class":1260},[1050,5318,2335],{"class":1088},[1050,5320,5321],{"class":1052,"line":1205},[1050,5322,1176],{"class":1067},[1050,5324,5325,5327,5329,5331,5333,5335,5337],{"class":1052,"line":1211},[1050,5326,1182],{"class":1148},[1050,5328,1282],{"class":1185},[1050,5330,1130],{"class":1088},[1050,5332,1192],{"class":1191},[1050,5334,1195],{"class":1088},[1050,5336,1291],{"class":1198},[1050,5338,1202],{"class":1088},[1050,5340,5341,5343,5345,5347,5349,5351,5353,5355],{"class":1052,"line":1217},[1050,5342,1352],{"class":1067},[1050,5344,1118],{"class":1088},[1050,5346,1357],{"class":1225},[1050,5348,1089],{"class":1088},[1050,5350,1232],{"class":1126},[1050,5352,1130],{"class":1088},[1050,5354,1366],{"class":1126},[1050,5356,1137],{"class":1088},[1050,5358,5359],{"class":1052,"line":1222},[1050,5360,1340],{"class":1067},[1050,5362,5363],{"class":1052,"line":1241},[1050,5364,5365],{"class":1056},"        # Track activity after successful response\n",[1050,5367,5368,5370,5373,5375,5377,5379,5382,5385,5387],{"class":1052,"line":1247},[1050,5369,2005],{"class":1063},[1050,5371,5372],{"class":1088}," (",[1050,5374,2013],{"class":1067},[1050,5376,1089],{"class":1088},[1050,5378,3159],{"class":1231},[1050,5380,5381],{"class":1409}," \u003C",[1050,5383,5384],{"class":2331}," 400",[1050,5386,3542],{"class":1409},[1050,5388,5389],{"class":1067}," \n",[1050,5391,5392,5394,5396,5399,5401,5403],{"class":1052,"line":1272},[1050,5393,3971],{"class":1225},[1050,5395,1089],{"class":1088},[1050,5397,5398],{"class":1126},"should_track_activity",[1050,5400,1130],{"class":1088},[1050,5402,1366],{"class":1126},[1050,5404,5405],{"class":1088},")):\n",[1050,5407,5408,5410,5412,5415,5417,5419,5421,5423],{"class":1052,"line":1277},[1050,5409,3971],{"class":1225},[1050,5411,1089],{"class":1088},[1050,5413,5414],{"class":1126},"track_user_activity",[1050,5416,1130],{"class":1088},[1050,5418,1366],{"class":1126},[1050,5420,1195],{"class":1088},[1050,5422,1631],{"class":1126},[1050,5424,1137],{"class":1088},[1050,5426,5427],{"class":1052,"line":1296},[1050,5428,1340],{"class":1067},[1050,5430,5431,5433],{"class":1052,"line":1301},[1050,5432,1478],{"class":1063},[1050,5434,1481],{"class":1067},[1050,5436,5437],{"class":1052,"line":1307},[1050,5438,1176],{"class":1067},[1050,5440,5441,5443,5446,5448,5450,5452,5454],{"class":1052,"line":1312},[1050,5442,1182],{"class":1148},[1050,5444,5445],{"class":1644}," should_track_activity",[1050,5447,1130],{"class":1088},[1050,5449,1192],{"class":1191},[1050,5451,1195],{"class":1088},[1050,5453,1291],{"class":1198},[1050,5455,1202],{"class":1088},[1050,5457,5458,5460,5463],{"class":1052,"line":1318},[1050,5459,1660],{"class":1162},[1050,5461,5462],{"class":1166},"Determine if activity should be tracked",[1050,5464,1170],{"class":1162},[1050,5466,5467],{"class":1052,"line":1337},[1050,5468,5469],{"class":1056},"        # Don't track static files\n",[1050,5471,5472,5474,5476,5478,5480,5482,5484,5486,5488,5490,5492,5494,5496,5498,5500,5502,5504,5506],{"class":1052,"line":1343},[1050,5473,2005],{"class":1063},[1050,5475,4033],{"class":1185},[1050,5477,1130],{"class":1088},[1050,5479,1366],{"class":1126},[1050,5481,1089],{"class":1088},[1050,5483,1752],{"class":1231},[1050,5485,1089],{"class":1088},[1050,5487,3528],{"class":1126},[1050,5489,1130],{"class":1088},[1050,5491,1752],{"class":1126},[1050,5493,1931],{"class":1088},[1050,5495,4054],{"class":1063},[1050,5497,4057],{"class":1126},[1050,5499,3501],{"class":1063},[1050,5501,1357],{"class":1225},[1050,5503,1089],{"class":1088},[1050,5505,5287],{"class":1231},[1050,5507,1202],{"class":1088},[1050,5509,5510,5512],{"class":1052,"line":1349},[1050,5511,4229],{"class":1063},[1050,5513,4232],{"class":1849},[1050,5515,5516],{"class":1052,"line":1371},[1050,5517,1340],{"class":1067},[1050,5519,5520],{"class":1052,"line":1376},[1050,5521,5522],{"class":1056},"        # Only track authenticated users\n",[1050,5524,5525,5527,5529,5531,5533,5535,5537,5539,5541,5543,5545,5548,5551,5553,5555,5557,5559,5561,5564],{"class":1052,"line":1382},[1050,5526,2005],{"class":1063},[1050,5528,4219],{"class":1409},[1050,5530,2008],{"class":1185},[1050,5532,1130],{"class":1088},[1050,5534,1366],{"class":1126},[1050,5536,1195],{"class":1088},[1050,5538,1915],{"class":1260},[1050,5540,2873],{"class":1264},[1050,5542,1436],{"class":1260},[1050,5544,1931],{"class":1088},[1050,5546,5547],{"class":1409}," or",[1050,5549,5550],{"class":1185}," isinstance",[1050,5552,1130],{"class":1088},[1050,5554,1366],{"class":1126},[1050,5556,1089],{"class":1088},[1050,5558,2873],{"class":1231},[1050,5560,1195],{"class":1088},[1050,5562,5563],{"class":1126}," AnonymousUser",[1050,5565,1202],{"class":1088},[1050,5567,5568,5570],{"class":1052,"line":1398},[1050,5569,4229],{"class":1063},[1050,5571,4232],{"class":1849},[1050,5573,5574],{"class":1052,"line":1416},[1050,5575,1340],{"class":1067},[1050,5577,5578],{"class":1052,"line":1421},[1050,5579,5580],{"class":1056},"        # Only track specific paths or all if no specific paths defined\n",[1050,5582,5583,5585,5587,5589,5591],{"class":1052,"line":1427},[1050,5584,2005],{"class":1063},[1050,5586,1357],{"class":1225},[1050,5588,1089],{"class":1088},[1050,5590,5247],{"class":1231},[1050,5592,1156],{"class":1088},[1050,5594,5595,5597,5599,5601,5603,5605,5607,5609,5611,5613,5615,5617,5619,5621,5623,5625,5627,5629],{"class":1052,"line":1470},[1050,5596,4229],{"class":1063},[1050,5598,4033],{"class":1185},[1050,5600,1130],{"class":1088},[1050,5602,1366],{"class":1126},[1050,5604,1089],{"class":1088},[1050,5606,1752],{"class":1231},[1050,5608,1089],{"class":1088},[1050,5610,3528],{"class":1126},[1050,5612,1130],{"class":1088},[1050,5614,1752],{"class":1126},[1050,5616,1931],{"class":1088},[1050,5618,4054],{"class":1063},[1050,5620,4057],{"class":1126},[1050,5622,3501],{"class":1063},[1050,5624,1357],{"class":1225},[1050,5626,1089],{"class":1088},[1050,5628,5247],{"class":1231},[1050,5630,1137],{"class":1088},[1050,5632,5633],{"class":1052,"line":1475},[1050,5634,1340],{"class":1067},[1050,5636,5637,5639],{"class":1052,"line":1965},[1050,5638,1478],{"class":1063},[1050,5640,4381],{"class":1849},[1050,5642,5643],{"class":1052,"line":1970},[1050,5644,1176],{"class":1067},[1050,5646,5647,5649,5652,5654,5656,5658,5660,5662,5664],{"class":1052,"line":1992},[1050,5648,1182],{"class":1148},[1050,5650,5651],{"class":1644}," track_user_activity",[1050,5653,1130],{"class":1088},[1050,5655,1192],{"class":1191},[1050,5657,1195],{"class":1088},[1050,5659,1291],{"class":1198},[1050,5661,1195],{"class":1088},[1050,5663,1631],{"class":1198},[1050,5665,1202],{"class":1088},[1050,5667,5668,5670,5673],{"class":1052,"line":2002},[1050,5669,1660],{"class":1162},[1050,5671,5672],{"class":1166},"Record user activity",[1050,5674,1170],{"class":1162},[1050,5676,5677,5680],{"class":1052,"line":2027},[1050,5678,5679],{"class":1063},"        try",[1050,5681,1156],{"class":1088},[1050,5683,5684,5687,5689],{"class":1052,"line":2072},[1050,5685,5686],{"class":1067},"            activity_data ",[1050,5688,1118],{"class":1088},[1050,5690,2773],{"class":1088},[1050,5692,5693,5696,5698,5700,5702,5704,5706,5708],{"class":1052,"line":2077},[1050,5694,5695],{"class":1260},"                '",[1050,5697,2873],{"class":1264},[1050,5699,1436],{"class":1260},[1050,5701,2785],{"class":1088},[1050,5703,1291],{"class":1067},[1050,5705,1089],{"class":1088},[1050,5707,2873],{"class":1231},[1050,5709,2485],{"class":1088},[1050,5711,5712,5714,5716,5718,5720,5722,5724,5726],{"class":1052,"line":2084},[1050,5713,5695],{"class":1260},[1050,5715,1752],{"class":1264},[1050,5717,1436],{"class":1260},[1050,5719,2785],{"class":1088},[1050,5721,1291],{"class":1067},[1050,5723,1089],{"class":1088},[1050,5725,1752],{"class":1231},[1050,5727,2485],{"class":1088},[1050,5729,5730,5732,5734,5736,5738,5740,5742,5744],{"class":1052,"line":2089},[1050,5731,5695],{"class":1260},[1050,5733,2809],{"class":1264},[1050,5735,1436],{"class":1260},[1050,5737,2785],{"class":1088},[1050,5739,1291],{"class":1067},[1050,5741,1089],{"class":1088},[1050,5743,2809],{"class":1231},[1050,5745,2485],{"class":1088},[1050,5747,5748,5750,5752,5754,5756,5758,5760,5762,5764,5766],{"class":1052,"line":2111},[1050,5749,5695],{"class":1260},[1050,5751,2927],{"class":1264},[1050,5753,1436],{"class":1260},[1050,5755,2785],{"class":1088},[1050,5757,1357],{"class":1225},[1050,5759,1089],{"class":1088},[1050,5761,2938],{"class":1126},[1050,5763,1130],{"class":1088},[1050,5765,1366],{"class":1126},[1050,5767,2866],{"class":1088},[1050,5769,5770,5772,5774,5776,5778,5780,5782,5784,5786,5788,5790,5792,5794,5796,5798,5800],{"class":1052,"line":2121},[1050,5771,5695],{"class":1260},[1050,5773,2951],{"class":1264},[1050,5775,1436],{"class":1260},[1050,5777,2785],{"class":1088},[1050,5779,1291],{"class":1067},[1050,5781,1089],{"class":1088},[1050,5783,2962],{"class":2862},[1050,5785,1089],{"class":1088},[1050,5787,2967],{"class":1126},[1050,5789,1130],{"class":1088},[1050,5791,1436],{"class":1260},[1050,5793,2974],{"class":1264},[1050,5795,1436],{"class":1260},[1050,5797,1195],{"class":1088},[1050,5799,2981],{"class":1260},[1050,5801,2866],{"class":1088},[1050,5803,5804,5806,5808,5810,5812,5814,5816,5818],{"class":1052,"line":2127},[1050,5805,5695],{"class":1260},[1050,5807,2780],{"class":1264},[1050,5809,1436],{"class":1260},[1050,5811,2785],{"class":1088},[1050,5813,2788],{"class":1067},[1050,5815,1089],{"class":1088},[1050,5817,2793],{"class":1126},[1050,5819,2802],{"class":1088},[1050,5821,5822,5824,5826,5828,5830,5832,5834],{"class":1052,"line":2169},[1050,5823,5695],{"class":1260},[1050,5825,3159],{"class":1264},[1050,5827,1436],{"class":1260},[1050,5829,2785],{"class":1088},[1050,5831,1631],{"class":1067},[1050,5833,1089],{"class":1088},[1050,5835,5836],{"class":1231},"status_code\n",[1050,5838,5839],{"class":1052,"line":2174},[1050,5840,5841],{"class":1088},"            }\n",[1050,5843,5844],{"class":1052,"line":2180},[1050,5845,5846],{"class":1067},"            \n",[1050,5848,5849],{"class":1052,"line":2201},[1050,5850,5851],{"class":1056},"            # Create activity record\n",[1050,5853,5854,5857,5859,5862,5864,5867,5869,5872,5875],{"class":1052,"line":2228},[1050,5855,5856],{"class":1067},"            UserActivity",[1050,5858,1089],{"class":1088},[1050,5860,5861],{"class":1231},"objects",[1050,5863,1089],{"class":1088},[1050,5865,5866],{"class":1126},"create",[1050,5868,1130],{"class":1088},[1050,5870,5871],{"class":1409},"**",[1050,5873,5874],{"class":1126},"activity_data",[1050,5876,1137],{"class":1088},[1050,5878,5879],{"class":1052,"line":2259},[1050,5880,5846],{"class":1067},[1050,5882,5883],{"class":1052,"line":2264},[1050,5884,5885],{"class":1056},"            # Update user's last activity timestamp\n",[1050,5887,5888,5891,5893,5895,5897,5900,5902,5904,5906,5908],{"class":1052,"line":2271},[1050,5889,5890],{"class":1067},"            request",[1050,5892,1089],{"class":1088},[1050,5894,2873],{"class":1231},[1050,5896,1089],{"class":1088},[1050,5898,5899],{"class":1231},"last_activity",[1050,5901,1235],{"class":1088},[1050,5903,2788],{"class":1067},[1050,5905,1089],{"class":1088},[1050,5907,2793],{"class":1126},[1050,5909,1334],{"class":1088},[1050,5911,5912,5914,5916,5918,5920,5923,5925,5928,5931,5933,5935,5937],{"class":1052,"line":2276},[1050,5913,5890],{"class":1067},[1050,5915,1089],{"class":1088},[1050,5917,2873],{"class":1231},[1050,5919,1089],{"class":1088},[1050,5921,5922],{"class":1126},"save",[1050,5924,1130],{"class":1088},[1050,5926,5927],{"class":4634},"update_fields",[1050,5929,5930],{"class":1088},"=[",[1050,5932,1436],{"class":1260},[1050,5934,5899],{"class":1264},[1050,5936,1436],{"class":1260},[1050,5938,4429],{"class":1088},[1050,5940,5941],{"class":1052,"line":2290},[1050,5942,5846],{"class":1067},[1050,5944,5945,5948,5951,5954,5957],{"class":1052,"line":2300},[1050,5946,5947],{"class":1063},"        except",[1050,5949,5950],{"class":2314}," Exception",[1050,5952,5953],{"class":1063}," as",[1050,5955,5956],{"class":1067}," e",[1050,5958,1156],{"class":1088},[1050,5960,5961],{"class":1052,"line":2309},[1050,5962,5963],{"class":1056},"            # Log error but don't break the request\n",[1050,5965,5966,5969,5971,5973,5975,5977,5980,5982,5985,5987,5989],{"class":1052,"line":3319},[1050,5967,5968],{"class":1067},"            logger",[1050,5970,1089],{"class":1088},[1050,5972,1894],{"class":1126},[1050,5974,1130],{"class":1088},[1050,5976,1727],{"class":1148},[1050,5978,5979],{"class":1264},"\"Error tracking user activity: ",[1050,5981,1455],{"class":1454},[1050,5983,5984],{"class":1126},"e",[1050,5986,1464],{"class":1454},[1050,5988,1261],{"class":1264},[1050,5990,1137],{"class":1088},[1050,5992,5993],{"class":1052,"line":3329},[1050,5994,1176],{"class":1067},[1050,5996,5997,5999,6001,6003,6005,6007,6009],{"class":1052,"line":3358},[1050,5998,1182],{"class":1148},[1050,6000,3306],{"class":1644},[1050,6002,1130],{"class":1088},[1050,6004,1192],{"class":1191},[1050,6006,1195],{"class":1088},[1050,6008,1291],{"class":1198},[1050,6010,1202],{"class":1088},[1050,6012,6013,6015,6017],{"class":1052,"line":3368},[1050,6014,1660],{"class":1162},[1050,6016,5015],{"class":1166},[1050,6018,1170],{"class":1162},[1050,6020,6021,6023,6025,6027,6029,6031,6033,6035,6037,6039,6041,6043],{"class":1052,"line":3399},[1050,6022,3332],{"class":1067},[1050,6024,1118],{"class":1088},[1050,6026,1291],{"class":1067},[1050,6028,1089],{"class":1088},[1050,6030,2962],{"class":2862},[1050,6032,1089],{"class":1088},[1050,6034,2967],{"class":1126},[1050,6036,1130],{"class":1088},[1050,6038,1436],{"class":1260},[1050,6040,3351],{"class":1264},[1050,6042,1436],{"class":1260},[1050,6044,1137],{"class":1088},[1050,6046,6047,6049,6051],{"class":1052,"line":3407},[1050,6048,2005],{"class":1063},[1050,6050,3363],{"class":1067},[1050,6052,1156],{"class":1088},[1050,6054,6055,6057,6059,6061,6063,6065,6067,6069,6071,6073,6075,6077],{"class":1052,"line":3435},[1050,6056,3371],{"class":1067},[1050,6058,1118],{"class":1088},[1050,6060,3363],{"class":1067},[1050,6062,1089],{"class":1088},[1050,6064,3380],{"class":1126},[1050,6066,1130],{"class":1088},[1050,6068,1436],{"class":1260},[1050,6070,1195],{"class":1264},[1050,6072,1436],{"class":1260},[1050,6074,3391],{"class":1088},[1050,6076,3394],{"class":2331},[1050,6078,2335],{"class":1088},[1050,6080,6081,6083],{"class":1052,"line":3443},[1050,6082,3402],{"class":1063},[1050,6084,1156],{"class":1088},[1050,6086,6087,6089,6091,6093,6095,6097,6099,6101,6103,6105,6107,6109],{"class":1052,"line":3448},[1050,6088,3371],{"class":1067},[1050,6090,1118],{"class":1088},[1050,6092,1291],{"class":1067},[1050,6094,1089],{"class":1088},[1050,6096,2962],{"class":2862},[1050,6098,1089],{"class":1088},[1050,6100,2967],{"class":1126},[1050,6102,1130],{"class":1088},[1050,6104,1436],{"class":1260},[1050,6106,3428],{"class":1264},[1050,6108,1436],{"class":1260},[1050,6110,1137],{"class":1088},[1050,6112,6113,6115],{"class":1052,"line":3466},[1050,6114,1478],{"class":1063},[1050,6116,3440],{"class":1067},[1050,6118,6119],{"class":1052,"line":3476},[1050,6120,1109],{"emptyLinePlaceholder":1108},[1050,6122,6123],{"class":1052,"line":3487},[1050,6124,6125],{"class":1056},"# models.py\n",[1050,6127,6128,6130,6133,6135,6139,6141,6144],{"class":1052,"line":3518},[1050,6129,1149],{"class":1148},[1050,6131,6132],{"class":1152}," UserActivity",[1050,6134,1130],{"class":1088},[1050,6136,6138],{"class":6137},"sYn-s","models",[1050,6140,1089],{"class":1088},[1050,6142,6143],{"class":6137},"Model",[1050,6145,1202],{"class":1088},[1050,6147,6148,6150,6153],{"class":1052,"line":3562},[1050,6149,1163],{"class":1162},[1050,6151,6152],{"class":1166},"Model to store user activity",[1050,6154,1170],{"class":1162},[1050,6156,6157,6160,6162,6165,6167,6170,6172,6175,6177,6180,6182,6184,6186,6189],{"class":1052,"line":3608},[1050,6158,6159],{"class":1067},"    user ",[1050,6161,1118],{"class":1088},[1050,6163,6164],{"class":1067}," models",[1050,6166,1089],{"class":1088},[1050,6168,6169],{"class":1126},"ForeignKey",[1050,6171,1130],{"class":1088},[1050,6173,6174],{"class":1126},"User",[1050,6176,1195],{"class":1088},[1050,6178,6179],{"class":4634}," on_delete",[1050,6181,1118],{"class":1088},[1050,6183,6138],{"class":1126},[1050,6185,1089],{"class":1088},[1050,6187,6188],{"class":2862},"CASCADE",[1050,6190,1137],{"class":1088},[1050,6192,6193,6196,6198,6200,6202,6205,6207,6210,6212,6215],{"class":1052,"line":3626},[1050,6194,6195],{"class":1067},"    path ",[1050,6197,1118],{"class":1088},[1050,6199,6164],{"class":1067},[1050,6201,1089],{"class":1088},[1050,6203,6204],{"class":1126},"CharField",[1050,6206,1130],{"class":1088},[1050,6208,6209],{"class":4634},"max_length",[1050,6211,1118],{"class":1088},[1050,6213,6214],{"class":2331},"255",[1050,6216,1137],{"class":1088},[1050,6218,6219,6222,6224,6226,6228,6230,6232,6234,6236,6239],{"class":1052,"line":4695},[1050,6220,6221],{"class":1067},"    method ",[1050,6223,1118],{"class":1088},[1050,6225,6164],{"class":1067},[1050,6227,1089],{"class":1088},[1050,6229,6204],{"class":1126},[1050,6231,1130],{"class":1088},[1050,6233,6209],{"class":4634},[1050,6235,1118],{"class":1088},[1050,6237,6238],{"class":2331},"10",[1050,6240,1137],{"class":1088},[1050,6242,6243,6246,6248,6250,6252,6255],{"class":1052,"line":4700},[1050,6244,6245],{"class":1067},"    ip_address ",[1050,6247,1118],{"class":1088},[1050,6249,6164],{"class":1067},[1050,6251,1089],{"class":1088},[1050,6253,6254],{"class":1126},"GenericIPAddressField",[1050,6256,1334],{"class":1088},[1050,6258,6259,6262,6264,6266,6268,6271],{"class":1052,"line":4722},[1050,6260,6261],{"class":1067},"    user_agent ",[1050,6263,1118],{"class":1088},[1050,6265,6164],{"class":1067},[1050,6267,1089],{"class":1088},[1050,6269,6270],{"class":1126},"TextField",[1050,6272,1334],{"class":1088},[1050,6274,6275,6278,6280,6282,6284,6287],{"class":1052,"line":4732},[1050,6276,6277],{"class":1067},"    timestamp ",[1050,6279,1118],{"class":1088},[1050,6281,6164],{"class":1067},[1050,6283,1089],{"class":1088},[1050,6285,6286],{"class":1126},"DateTimeField",[1050,6288,1334],{"class":1088},[1050,6290,6291,6294,6296,6298,6300,6303],{"class":1052,"line":4751},[1050,6292,6293],{"class":1067},"    status_code ",[1050,6295,1118],{"class":1088},[1050,6297,6164],{"class":1067},[1050,6299,1089],{"class":1088},[1050,6301,6302],{"class":1126},"IntegerField",[1050,6304,1334],{"class":1088},[1050,6306,6307],{"class":1052,"line":4762},[1050,6308,1176],{"class":1067},[1050,6310,6311,6314,6317],{"class":1052,"line":4768},[1050,6312,6313],{"class":1148},"    class",[1050,6315,6316],{"class":1152}," Meta",[1050,6318,1156],{"class":1088},[1050,6320,6321,6324,6326,6328,6330,6333,6335],{"class":1052,"line":4773},[1050,6322,6323],{"class":1067},"        ordering ",[1050,6325,1118],{"class":1088},[1050,6327,5252],{"class":1088},[1050,6329,1436],{"class":1260},[1050,6331,6332],{"class":1264},"-timestamp",[1050,6334,1436],{"class":1260},[1050,6336,2335],{"class":1088},[1050,6338,6339,6342,6344],{"class":1052,"line":4792},[1050,6340,6341],{"class":1067},"        indexes ",[1050,6343,1118],{"class":1088},[1050,6345,2472],{"class":1088},[1050,6347,6348,6351,6353,6356,6358,6361,6363,6365,6367,6369,6371,6373,6375,6377],{"class":1052,"line":4841},[1050,6349,6350],{"class":1067},"            models",[1050,6352,1089],{"class":1088},[1050,6354,6355],{"class":1126},"Index",[1050,6357,1130],{"class":1088},[1050,6359,6360],{"class":4634},"fields",[1050,6362,5930],{"class":1088},[1050,6364,1436],{"class":1260},[1050,6366,2873],{"class":1264},[1050,6368,1436],{"class":1260},[1050,6370,1195],{"class":1088},[1050,6372,1915],{"class":1260},[1050,6374,2780],{"class":1264},[1050,6376,1436],{"class":1260},[1050,6378,6379],{"class":1088},"]),\n",[1050,6381,6382,6384,6386,6388,6390,6392,6394,6396,6398,6400],{"class":1052,"line":4864},[1050,6383,6350],{"class":1067},[1050,6385,1089],{"class":1088},[1050,6387,6355],{"class":1126},[1050,6389,1130],{"class":1088},[1050,6391,6360],{"class":4634},[1050,6393,5930],{"class":1088},[1050,6395,1436],{"class":1260},[1050,6397,2780],{"class":1264},[1050,6399,1436],{"class":1260},[1050,6401,6379],{"class":1088},[1050,6403,6404],{"class":1052,"line":4869},[1050,6405,2511],{"class":1088},[1035,6407,6409],{"id":6408},"content-security-policy-middleware","Content Security Policy Middleware",[1040,6411,6413],{"className":1042,"code":6412,"language":1044,"meta":1045,"style":1045},"class CSPMiddleware:\n    \"\"\"Content Security Policy middleware\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n        self.default_policy = {\n            'default-src': [\"'self'\"],\n            'script-src': [\"'self'\", \"'unsafe-inline'\"],\n            'style-src': [\"'self'\", \"'unsafe-inline'\"],\n            'img-src': [\"'self'\", \"data:\", \"https:\"],\n            'font-src': [\"'self'\"],\n            'connect-src': [\"'self'\"],\n            'frame-src': [\"'none'\"],\n            'object-src': [\"'none'\"],\n            'base-uri': [\"'self'\"],\n            'form-action': [\"'self'\"]\n        }\n    \n    def __call__(self, request):\n        response = self.get_response(request)\n        \n        # Add CSP header\n        csp_header = self.build_csp_header(request)\n        response['Content-Security-Policy'] = csp_header\n        \n        return response\n    \n    def build_csp_header(self, request):\n        \"\"\"Build CSP header based on request\"\"\"\n        policy = self.default_policy.copy()\n        \n        # Customize policy based on request path\n        if request.path.startswith('/admin/'):\n            # More permissive policy for admin\n            policy['script-src'].append(\"'unsafe-eval'\")\n        \n        elif request.path.startswith('/api/'):\n            # Strict policy for API\n            policy['script-src'] = [\"'none'\"]\n            policy['style-src'] = [\"'none'\"]\n        \n        # Build header string\n        directives = []\n        for directive, sources in policy.items():\n            sources_str = ' '.join(sources)\n            directives.append(f\"{directive} {sources_str}\")\n        \n        return '; '.join(directives)\n",[1047,6414,6415,6424,6433,6437,6453,6465,6478,6501,6532,6561,6600,6621,6642,6664,6685,6706,6727,6731,6735,6751,6769,6773,6778,6798,6818,6822,6828,6832,6849,6858,6878,6882,6887,6912,6917,6947,6951,6976,6981,7007,7033,7037,7042,7052,7075,7098,7131,7135],{"__ignoreMap":1045},[1050,6416,6417,6419,6422],{"class":1052,"line":1053},[1050,6418,1149],{"class":1148},[1050,6420,6421],{"class":1152}," CSPMiddleware",[1050,6423,1156],{"class":1088},[1050,6425,6426,6428,6431],{"class":1052,"line":1060},[1050,6427,1163],{"class":1162},[1050,6429,6430],{"class":1166},"Content Security Policy middleware",[1050,6432,1170],{"class":1162},[1050,6434,6435],{"class":1052,"line":1071},[1050,6436,1176],{"class":1067},[1050,6438,6439,6441,6443,6445,6447,6449,6451],{"class":1052,"line":1079},[1050,6440,1182],{"class":1148},[1050,6442,1186],{"class":1185},[1050,6444,1130],{"class":1088},[1050,6446,1192],{"class":1191},[1050,6448,1195],{"class":1088},[1050,6450,1199],{"class":1198},[1050,6452,1202],{"class":1088},[1050,6454,6455,6457,6459,6461,6463],{"class":1052,"line":1105},[1050,6456,1226],{"class":1225},[1050,6458,1089],{"class":1088},[1050,6460,1232],{"class":1231},[1050,6462,1235],{"class":1088},[1050,6464,1238],{"class":1067},[1050,6466,6467,6469,6471,6474,6476],{"class":1052,"line":1112},[1050,6468,1226],{"class":1225},[1050,6470,1089],{"class":1088},[1050,6472,6473],{"class":1231},"default_policy",[1050,6475,1235],{"class":1088},[1050,6477,2773],{"class":1088},[1050,6479,6480,6482,6485,6487,6489,6491,6493,6496,6498],{"class":1052,"line":1140},[1050,6481,2477],{"class":1260},[1050,6483,6484],{"class":1264},"default-src",[1050,6486,1436],{"class":1260},[1050,6488,2785],{"class":1088},[1050,6490,5252],{"class":1088},[1050,6492,1261],{"class":1260},[1050,6494,6495],{"class":1264},"'self'",[1050,6497,1261],{"class":1260},[1050,6499,6500],{"class":1088},"],\n",[1050,6502,6503,6505,6508,6510,6512,6514,6516,6518,6520,6522,6525,6528,6530],{"class":1052,"line":1145},[1050,6504,2477],{"class":1260},[1050,6506,6507],{"class":1264},"script-src",[1050,6509,1436],{"class":1260},[1050,6511,2785],{"class":1088},[1050,6513,5252],{"class":1088},[1050,6515,1261],{"class":1260},[1050,6517,6495],{"class":1264},[1050,6519,1261],{"class":1260},[1050,6521,1195],{"class":1088},[1050,6523,6524],{"class":1260}," \"",[1050,6526,6527],{"class":1264},"'unsafe-inline'",[1050,6529,1261],{"class":1260},[1050,6531,6500],{"class":1088},[1050,6533,6534,6536,6539,6541,6543,6545,6547,6549,6551,6553,6555,6557,6559],{"class":1052,"line":1159},[1050,6535,2477],{"class":1260},[1050,6537,6538],{"class":1264},"style-src",[1050,6540,1436],{"class":1260},[1050,6542,2785],{"class":1088},[1050,6544,5252],{"class":1088},[1050,6546,1261],{"class":1260},[1050,6548,6495],{"class":1264},[1050,6550,1261],{"class":1260},[1050,6552,1195],{"class":1088},[1050,6554,6524],{"class":1260},[1050,6556,6527],{"class":1264},[1050,6558,1261],{"class":1260},[1050,6560,6500],{"class":1088},[1050,6562,6563,6565,6568,6570,6572,6574,6576,6578,6580,6582,6584,6587,6589,6591,6593,6596,6598],{"class":1052,"line":1173},[1050,6564,2477],{"class":1260},[1050,6566,6567],{"class":1264},"img-src",[1050,6569,1436],{"class":1260},[1050,6571,2785],{"class":1088},[1050,6573,5252],{"class":1088},[1050,6575,1261],{"class":1260},[1050,6577,6495],{"class":1264},[1050,6579,1261],{"class":1260},[1050,6581,1195],{"class":1088},[1050,6583,6524],{"class":1260},[1050,6585,6586],{"class":1264},"data:",[1050,6588,1261],{"class":1260},[1050,6590,1195],{"class":1088},[1050,6592,6524],{"class":1260},[1050,6594,6595],{"class":1264},"https:",[1050,6597,1261],{"class":1260},[1050,6599,6500],{"class":1088},[1050,6601,6602,6604,6607,6609,6611,6613,6615,6617,6619],{"class":1052,"line":1179},[1050,6603,2477],{"class":1260},[1050,6605,6606],{"class":1264},"font-src",[1050,6608,1436],{"class":1260},[1050,6610,2785],{"class":1088},[1050,6612,5252],{"class":1088},[1050,6614,1261],{"class":1260},[1050,6616,6495],{"class":1264},[1050,6618,1261],{"class":1260},[1050,6620,6500],{"class":1088},[1050,6622,6623,6625,6628,6630,6632,6634,6636,6638,6640],{"class":1052,"line":1205},[1050,6624,2477],{"class":1260},[1050,6626,6627],{"class":1264},"connect-src",[1050,6629,1436],{"class":1260},[1050,6631,2785],{"class":1088},[1050,6633,5252],{"class":1088},[1050,6635,1261],{"class":1260},[1050,6637,6495],{"class":1264},[1050,6639,1261],{"class":1260},[1050,6641,6500],{"class":1088},[1050,6643,6644,6646,6649,6651,6653,6655,6657,6660,6662],{"class":1052,"line":1211},[1050,6645,2477],{"class":1260},[1050,6647,6648],{"class":1264},"frame-src",[1050,6650,1436],{"class":1260},[1050,6652,2785],{"class":1088},[1050,6654,5252],{"class":1088},[1050,6656,1261],{"class":1260},[1050,6658,6659],{"class":1264},"'none'",[1050,6661,1261],{"class":1260},[1050,6663,6500],{"class":1088},[1050,6665,6666,6668,6671,6673,6675,6677,6679,6681,6683],{"class":1052,"line":1217},[1050,6667,2477],{"class":1260},[1050,6669,6670],{"class":1264},"object-src",[1050,6672,1436],{"class":1260},[1050,6674,2785],{"class":1088},[1050,6676,5252],{"class":1088},[1050,6678,1261],{"class":1260},[1050,6680,6659],{"class":1264},[1050,6682,1261],{"class":1260},[1050,6684,6500],{"class":1088},[1050,6686,6687,6689,6692,6694,6696,6698,6700,6702,6704],{"class":1052,"line":1222},[1050,6688,2477],{"class":1260},[1050,6690,6691],{"class":1264},"base-uri",[1050,6693,1436],{"class":1260},[1050,6695,2785],{"class":1088},[1050,6697,5252],{"class":1088},[1050,6699,1261],{"class":1260},[1050,6701,6495],{"class":1264},[1050,6703,1261],{"class":1260},[1050,6705,6500],{"class":1088},[1050,6707,6708,6710,6713,6715,6717,6719,6721,6723,6725],{"class":1052,"line":1241},[1050,6709,2477],{"class":1260},[1050,6711,6712],{"class":1264},"form-action",[1050,6714,1436],{"class":1260},[1050,6716,2785],{"class":1088},[1050,6718,5252],{"class":1088},[1050,6720,1261],{"class":1260},[1050,6722,6495],{"class":1264},[1050,6724,1261],{"class":1260},[1050,6726,2335],{"class":1088},[1050,6728,6729],{"class":1052,"line":1247},[1050,6730,3085],{"class":1088},[1050,6732,6733],{"class":1052,"line":1272},[1050,6734,1176],{"class":1067},[1050,6736,6737,6739,6741,6743,6745,6747,6749],{"class":1052,"line":1277},[1050,6738,1182],{"class":1148},[1050,6740,1282],{"class":1185},[1050,6742,1130],{"class":1088},[1050,6744,1192],{"class":1191},[1050,6746,1195],{"class":1088},[1050,6748,1291],{"class":1198},[1050,6750,1202],{"class":1088},[1050,6752,6753,6755,6757,6759,6761,6763,6765,6767],{"class":1052,"line":1296},[1050,6754,1352],{"class":1067},[1050,6756,1118],{"class":1088},[1050,6758,1357],{"class":1225},[1050,6760,1089],{"class":1088},[1050,6762,1232],{"class":1126},[1050,6764,1130],{"class":1088},[1050,6766,1366],{"class":1126},[1050,6768,1137],{"class":1088},[1050,6770,6771],{"class":1052,"line":1301},[1050,6772,1340],{"class":1067},[1050,6774,6775],{"class":1052,"line":1307},[1050,6776,6777],{"class":1056},"        # Add CSP header\n",[1050,6779,6780,6783,6785,6787,6789,6792,6794,6796],{"class":1052,"line":1312},[1050,6781,6782],{"class":1067},"        csp_header ",[1050,6784,1118],{"class":1088},[1050,6786,1357],{"class":1225},[1050,6788,1089],{"class":1088},[1050,6790,6791],{"class":1126},"build_csp_header",[1050,6793,1130],{"class":1088},[1050,6795,1366],{"class":1126},[1050,6797,1137],{"class":1088},[1050,6799,6800,6802,6804,6806,6809,6811,6813,6815],{"class":1052,"line":1318},[1050,6801,1430],{"class":1067},[1050,6803,1433],{"class":1088},[1050,6805,1436],{"class":1260},[1050,6807,6808],{"class":1264},"Content-Security-Policy",[1050,6810,1436],{"class":1260},[1050,6812,1444],{"class":1088},[1050,6814,1235],{"class":1088},[1050,6816,6817],{"class":1067}," csp_header\n",[1050,6819,6820],{"class":1052,"line":1337},[1050,6821,1340],{"class":1067},[1050,6823,6824,6826],{"class":1052,"line":1343},[1050,6825,1478],{"class":1063},[1050,6827,1481],{"class":1067},[1050,6829,6830],{"class":1052,"line":1349},[1050,6831,1176],{"class":1067},[1050,6833,6834,6836,6839,6841,6843,6845,6847],{"class":1052,"line":1371},[1050,6835,1182],{"class":1148},[1050,6837,6838],{"class":1644}," build_csp_header",[1050,6840,1130],{"class":1088},[1050,6842,1192],{"class":1191},[1050,6844,1195],{"class":1088},[1050,6846,1291],{"class":1198},[1050,6848,1202],{"class":1088},[1050,6850,6851,6853,6856],{"class":1052,"line":1376},[1050,6852,1660],{"class":1162},[1050,6854,6855],{"class":1166},"Build CSP header based on request",[1050,6857,1170],{"class":1162},[1050,6859,6860,6863,6865,6867,6869,6871,6873,6876],{"class":1052,"line":1382},[1050,6861,6862],{"class":1067},"        policy ",[1050,6864,1118],{"class":1088},[1050,6866,1357],{"class":1225},[1050,6868,1089],{"class":1088},[1050,6870,6473],{"class":1231},[1050,6872,1089],{"class":1088},[1050,6874,6875],{"class":1126},"copy",[1050,6877,1334],{"class":1088},[1050,6879,6880],{"class":1052,"line":1398},[1050,6881,1340],{"class":1067},[1050,6883,6884],{"class":1052,"line":1416},[1050,6885,6886],{"class":1056},"        # Customize policy based on request path\n",[1050,6888,6889,6891,6893,6895,6897,6899,6901,6903,6905,6908,6910],{"class":1052,"line":1421},[1050,6890,2005],{"class":1063},[1050,6892,1291],{"class":1067},[1050,6894,1089],{"class":1088},[1050,6896,1752],{"class":1231},[1050,6898,1089],{"class":1088},[1050,6900,3528],{"class":1126},[1050,6902,1130],{"class":1088},[1050,6904,1436],{"class":1260},[1050,6906,6907],{"class":1264},"/admin/",[1050,6909,1436],{"class":1260},[1050,6911,1202],{"class":1088},[1050,6913,6914],{"class":1052,"line":1427},[1050,6915,6916],{"class":1056},"            # More permissive policy for admin\n",[1050,6918,6919,6922,6924,6926,6928,6930,6933,6936,6938,6940,6943,6945],{"class":1052,"line":1470},[1050,6920,6921],{"class":1067},"            policy",[1050,6923,1433],{"class":1088},[1050,6925,1436],{"class":1260},[1050,6927,6507],{"class":1264},[1050,6929,1436],{"class":1260},[1050,6931,6932],{"class":1088},"].",[1050,6934,6935],{"class":1126},"append",[1050,6937,1130],{"class":1088},[1050,6939,1261],{"class":1260},[1050,6941,6942],{"class":1264},"'unsafe-eval'",[1050,6944,1261],{"class":1260},[1050,6946,1137],{"class":1088},[1050,6948,6949],{"class":1052,"line":1475},[1050,6950,1340],{"class":1067},[1050,6952,6953,6956,6958,6960,6962,6964,6966,6968,6970,6972,6974],{"class":1052,"line":1965},[1050,6954,6955],{"class":1063},"        elif",[1050,6957,1291],{"class":1067},[1050,6959,1089],{"class":1088},[1050,6961,1752],{"class":1231},[1050,6963,1089],{"class":1088},[1050,6965,3528],{"class":1126},[1050,6967,1130],{"class":1088},[1050,6969,1436],{"class":1260},[1050,6971,3757],{"class":1264},[1050,6973,1436],{"class":1260},[1050,6975,1202],{"class":1088},[1050,6977,6978],{"class":1052,"line":1970},[1050,6979,6980],{"class":1056},"            # Strict policy for API\n",[1050,6982,6983,6985,6987,6989,6991,6993,6995,6997,6999,7001,7003,7005],{"class":1052,"line":1992},[1050,6984,6921],{"class":1067},[1050,6986,1433],{"class":1088},[1050,6988,1436],{"class":1260},[1050,6990,6507],{"class":1264},[1050,6992,1436],{"class":1260},[1050,6994,1444],{"class":1088},[1050,6996,1235],{"class":1088},[1050,6998,5252],{"class":1088},[1050,7000,1261],{"class":1260},[1050,7002,6659],{"class":1264},[1050,7004,1261],{"class":1260},[1050,7006,2335],{"class":1088},[1050,7008,7009,7011,7013,7015,7017,7019,7021,7023,7025,7027,7029,7031],{"class":1052,"line":2002},[1050,7010,6921],{"class":1067},[1050,7012,1433],{"class":1088},[1050,7014,1436],{"class":1260},[1050,7016,6538],{"class":1264},[1050,7018,1436],{"class":1260},[1050,7020,1444],{"class":1088},[1050,7022,1235],{"class":1088},[1050,7024,5252],{"class":1088},[1050,7026,1261],{"class":1260},[1050,7028,6659],{"class":1264},[1050,7030,1261],{"class":1260},[1050,7032,2335],{"class":1088},[1050,7034,7035],{"class":1052,"line":2027},[1050,7036,1340],{"class":1067},[1050,7038,7039],{"class":1052,"line":2072},[1050,7040,7041],{"class":1056},"        # Build header string\n",[1050,7043,7044,7047,7049],{"class":1052,"line":2077},[1050,7045,7046],{"class":1067},"        directives ",[1050,7048,1118],{"class":1088},[1050,7050,7051],{"class":1088}," []\n",[1050,7053,7054,7056,7059,7061,7064,7066,7069,7071,7073],{"class":1052,"line":2084},[1050,7055,3490],{"class":1063},[1050,7057,7058],{"class":1067}," directive",[1050,7060,1195],{"class":1088},[1050,7062,7063],{"class":1067}," sources ",[1050,7065,3501],{"class":1063},[1050,7067,7068],{"class":1067}," policy",[1050,7070,1089],{"class":1088},[1050,7072,3512],{"class":1126},[1050,7074,3515],{"class":1088},[1050,7076,7077,7080,7082,7084,7086,7088,7091,7093,7096],{"class":1052,"line":2089},[1050,7078,7079],{"class":1067},"            sources_str ",[1050,7081,1118],{"class":1088},[1050,7083,1915],{"class":1260},[1050,7085,1915],{"class":1260},[1050,7087,1089],{"class":1088},[1050,7089,7090],{"class":1126},"join",[1050,7092,1130],{"class":1088},[1050,7094,7095],{"class":1126},"sources",[1050,7097,1137],{"class":1088},[1050,7099,7100,7103,7105,7107,7109,7111,7113,7115,7118,7120,7122,7125,7127,7129],{"class":1052,"line":2111},[1050,7101,7102],{"class":1067},"            directives",[1050,7104,1089],{"class":1088},[1050,7106,6935],{"class":1126},[1050,7108,1130],{"class":1088},[1050,7110,1727],{"class":1148},[1050,7112,1261],{"class":1264},[1050,7114,1455],{"class":1454},[1050,7116,7117],{"class":1126},"directive",[1050,7119,1464],{"class":1454},[1050,7121,3764],{"class":1454},[1050,7123,7124],{"class":1126},"sources_str",[1050,7126,1464],{"class":1454},[1050,7128,1261],{"class":1264},[1050,7130,1137],{"class":1088},[1050,7132,7133],{"class":1052,"line":2121},[1050,7134,1340],{"class":1067},[1050,7136,7137,7139,7141,7144,7146,7148,7150,7152,7155],{"class":1052,"line":2127},[1050,7138,1478],{"class":1063},[1050,7140,1915],{"class":1260},[1050,7142,7143],{"class":1264},"; ",[1050,7145,1436],{"class":1260},[1050,7147,1089],{"class":1088},[1050,7149,7090],{"class":1126},[1050,7151,1130],{"class":1088},[1050,7153,7154],{"class":1126},"directives",[1050,7156,1137],{"class":1088},[1035,7158,7160],{"id":7159},"mobile-detection-middleware","Mobile Detection Middleware",[1040,7162,7164],{"className":1042,"code":7163,"language":1044,"meta":1045,"style":1045},"import re\n\nclass MobileDetectionMiddleware:\n    \"\"\"Detect mobile devices and add context\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n        \n        # Mobile user agent patterns\n        self.mobile_patterns = [\n            re.compile(r'Mobile', re.IGNORECASE),\n            re.compile(r'Android', re.IGNORECASE),\n            re.compile(r'iPhone', re.IGNORECASE),\n            re.compile(r'iPad', re.IGNORECASE),\n            re.compile(r'Windows Phone', re.IGNORECASE),\n            re.compile(r'BlackBerry', re.IGNORECASE),\n        ]\n        \n        # Tablet patterns\n        self.tablet_patterns = [\n            re.compile(r'iPad', re.IGNORECASE),\n            re.compile(r'Android.*Tablet', re.IGNORECASE),\n            re.compile(r'Kindle', re.IGNORECASE),\n        ]\n    \n    def __call__(self, request):\n        # Detect device type\n        user_agent = request.META.get('HTTP_USER_AGENT', '')\n        \n        request.is_mobile = self.is_mobile(user_agent)\n        request.is_tablet = self.is_tablet(user_agent)\n        request.is_desktop = not (request.is_mobile or request.is_tablet)\n        \n        # Add device info to request\n        request.device_info = {\n            'type': self.get_device_type(request),\n            'user_agent': user_agent,\n            'screen_size': self.estimate_screen_size(request)\n        }\n        \n        response = self.get_response(request)\n        \n        # Add device type header\n        response['X-Device-Type'] = request.device_info['type']\n        \n        return response\n    \n    def is_mobile(self, user_agent):\n        \"\"\"Check if user agent indicates mobile device\"\"\"\n        return any(pattern.search(user_agent) for pattern in self.mobile_patterns)\n    \n    def is_tablet(self, user_agent):\n        \"\"\"Check if user agent indicates tablet device\"\"\"\n        return any(pattern.search(user_agent) for pattern in self.tablet_patterns)\n    \n    def get_device_type(self, request):\n        \"\"\"Get device type string\"\"\"\n        if request.is_tablet:\n            return 'tablet'\n        elif request.is_mobile:\n            return 'mobile'\n        else:\n            return 'desktop'\n    \n    def estimate_screen_size(self, request):\n        \"\"\"Estimate screen size category\"\"\"\n        if request.is_mobile:\n            return 'small'\n        elif request.is_tablet:\n            return 'medium'\n        else:\n            return 'large'\n",[1047,7165,7166,7173,7177,7186,7195,7199,7215,7227,7231,7236,7249,7284,7313,7342,7371,7400,7429,7433,7437,7442,7455,7483,7521,7550,7554,7558,7574,7579,7610,7614,7637,7660,7691,7695,7700,7713,7737,7752,7776,7780,7784,7802,7806,7811,7844,7848,7854,7858,7875,7884,7921,7925,7942,7951,7985,7989,8006,8015,8027,8038,8050,8061,8067,8078,8082,8099,8108,8120,8131,8143,8154,8160],{"__ignoreMap":1045},[1050,7167,7168,7170],{"class":1052,"line":1053},[1050,7169,1064],{"class":1063},[1050,7171,7172],{"class":1067}," re\n",[1050,7174,7175],{"class":1052,"line":1060},[1050,7176,1109],{"emptyLinePlaceholder":1108},[1050,7178,7179,7181,7184],{"class":1052,"line":1071},[1050,7180,1149],{"class":1148},[1050,7182,7183],{"class":1152}," MobileDetectionMiddleware",[1050,7185,1156],{"class":1088},[1050,7187,7188,7190,7193],{"class":1052,"line":1079},[1050,7189,1163],{"class":1162},[1050,7191,7192],{"class":1166},"Detect mobile devices and add context",[1050,7194,1170],{"class":1162},[1050,7196,7197],{"class":1052,"line":1105},[1050,7198,1176],{"class":1067},[1050,7200,7201,7203,7205,7207,7209,7211,7213],{"class":1052,"line":1112},[1050,7202,1182],{"class":1148},[1050,7204,1186],{"class":1185},[1050,7206,1130],{"class":1088},[1050,7208,1192],{"class":1191},[1050,7210,1195],{"class":1088},[1050,7212,1199],{"class":1198},[1050,7214,1202],{"class":1088},[1050,7216,7217,7219,7221,7223,7225],{"class":1052,"line":1140},[1050,7218,1226],{"class":1225},[1050,7220,1089],{"class":1088},[1050,7222,1232],{"class":1231},[1050,7224,1235],{"class":1088},[1050,7226,1238],{"class":1067},[1050,7228,7229],{"class":1052,"line":1145},[1050,7230,1340],{"class":1067},[1050,7232,7233],{"class":1052,"line":1159},[1050,7234,7235],{"class":1056},"        # Mobile user agent patterns\n",[1050,7237,7238,7240,7242,7245,7247],{"class":1052,"line":1173},[1050,7239,1226],{"class":1225},[1050,7241,1089],{"class":1088},[1050,7243,7244],{"class":1231},"mobile_patterns",[1050,7246,1235],{"class":1088},[1050,7248,2472],{"class":1088},[1050,7250,7251,7254,7256,7259,7261,7264,7266,7270,7272,7274,7277,7279,7282],{"class":1052,"line":1179},[1050,7252,7253],{"class":1067},"            re",[1050,7255,1089],{"class":1088},[1050,7257,7258],{"class":1126},"compile",[1050,7260,1130],{"class":1088},[1050,7262,7263],{"class":1148},"r",[1050,7265,1436],{"class":1260},[1050,7267,7269],{"class":7268},"s27EL","Mobile",[1050,7271,1436],{"class":1260},[1050,7273,1195],{"class":1088},[1050,7275,7276],{"class":1126}," re",[1050,7278,1089],{"class":1088},[1050,7280,7281],{"class":2862},"IGNORECASE",[1050,7283,2866],{"class":1088},[1050,7285,7286,7288,7290,7292,7294,7296,7298,7301,7303,7305,7307,7309,7311],{"class":1052,"line":1205},[1050,7287,7253],{"class":1067},[1050,7289,1089],{"class":1088},[1050,7291,7258],{"class":1126},[1050,7293,1130],{"class":1088},[1050,7295,7263],{"class":1148},[1050,7297,1436],{"class":1260},[1050,7299,7300],{"class":7268},"Android",[1050,7302,1436],{"class":1260},[1050,7304,1195],{"class":1088},[1050,7306,7276],{"class":1126},[1050,7308,1089],{"class":1088},[1050,7310,7281],{"class":2862},[1050,7312,2866],{"class":1088},[1050,7314,7315,7317,7319,7321,7323,7325,7327,7330,7332,7334,7336,7338,7340],{"class":1052,"line":1211},[1050,7316,7253],{"class":1067},[1050,7318,1089],{"class":1088},[1050,7320,7258],{"class":1126},[1050,7322,1130],{"class":1088},[1050,7324,7263],{"class":1148},[1050,7326,1436],{"class":1260},[1050,7328,7329],{"class":7268},"iPhone",[1050,7331,1436],{"class":1260},[1050,7333,1195],{"class":1088},[1050,7335,7276],{"class":1126},[1050,7337,1089],{"class":1088},[1050,7339,7281],{"class":2862},[1050,7341,2866],{"class":1088},[1050,7343,7344,7346,7348,7350,7352,7354,7356,7359,7361,7363,7365,7367,7369],{"class":1052,"line":1217},[1050,7345,7253],{"class":1067},[1050,7347,1089],{"class":1088},[1050,7349,7258],{"class":1126},[1050,7351,1130],{"class":1088},[1050,7353,7263],{"class":1148},[1050,7355,1436],{"class":1260},[1050,7357,7358],{"class":7268},"iPad",[1050,7360,1436],{"class":1260},[1050,7362,1195],{"class":1088},[1050,7364,7276],{"class":1126},[1050,7366,1089],{"class":1088},[1050,7368,7281],{"class":2862},[1050,7370,2866],{"class":1088},[1050,7372,7373,7375,7377,7379,7381,7383,7385,7388,7390,7392,7394,7396,7398],{"class":1052,"line":1222},[1050,7374,7253],{"class":1067},[1050,7376,1089],{"class":1088},[1050,7378,7258],{"class":1126},[1050,7380,1130],{"class":1088},[1050,7382,7263],{"class":1148},[1050,7384,1436],{"class":1260},[1050,7386,7387],{"class":7268},"Windows Phone",[1050,7389,1436],{"class":1260},[1050,7391,1195],{"class":1088},[1050,7393,7276],{"class":1126},[1050,7395,1089],{"class":1088},[1050,7397,7281],{"class":2862},[1050,7399,2866],{"class":1088},[1050,7401,7402,7404,7406,7408,7410,7412,7414,7417,7419,7421,7423,7425,7427],{"class":1052,"line":1241},[1050,7403,7253],{"class":1067},[1050,7405,1089],{"class":1088},[1050,7407,7258],{"class":1126},[1050,7409,1130],{"class":1088},[1050,7411,7263],{"class":1148},[1050,7413,1436],{"class":1260},[1050,7415,7416],{"class":7268},"BlackBerry",[1050,7418,1436],{"class":1260},[1050,7420,1195],{"class":1088},[1050,7422,7276],{"class":1126},[1050,7424,1089],{"class":1088},[1050,7426,7281],{"class":2862},[1050,7428,2866],{"class":1088},[1050,7430,7431],{"class":1052,"line":1247},[1050,7432,2511],{"class":1088},[1050,7434,7435],{"class":1052,"line":1272},[1050,7436,1340],{"class":1067},[1050,7438,7439],{"class":1052,"line":1277},[1050,7440,7441],{"class":1056},"        # Tablet patterns\n",[1050,7443,7444,7446,7448,7451,7453],{"class":1052,"line":1296},[1050,7445,1226],{"class":1225},[1050,7447,1089],{"class":1088},[1050,7449,7450],{"class":1231},"tablet_patterns",[1050,7452,1235],{"class":1088},[1050,7454,2472],{"class":1088},[1050,7456,7457,7459,7461,7463,7465,7467,7469,7471,7473,7475,7477,7479,7481],{"class":1052,"line":1301},[1050,7458,7253],{"class":1067},[1050,7460,1089],{"class":1088},[1050,7462,7258],{"class":1126},[1050,7464,1130],{"class":1088},[1050,7466,7263],{"class":1148},[1050,7468,1436],{"class":1260},[1050,7470,7358],{"class":7268},[1050,7472,1436],{"class":1260},[1050,7474,1195],{"class":1088},[1050,7476,7276],{"class":1126},[1050,7478,1089],{"class":1088},[1050,7480,7281],{"class":2862},[1050,7482,2866],{"class":1088},[1050,7484,7485,7487,7489,7491,7493,7495,7497,7499,7502,7506,7509,7511,7513,7515,7517,7519],{"class":1052,"line":1307},[1050,7486,7253],{"class":1067},[1050,7488,1089],{"class":1088},[1050,7490,7258],{"class":1126},[1050,7492,1130],{"class":1088},[1050,7494,7263],{"class":1148},[1050,7496,1436],{"class":1260},[1050,7498,7300],{"class":7268},[1050,7500,1089],{"class":7501},"sRJPX",[1050,7503,7505],{"class":7504},"sGuSu","*",[1050,7507,7508],{"class":7268},"Tablet",[1050,7510,1436],{"class":1260},[1050,7512,1195],{"class":1088},[1050,7514,7276],{"class":1126},[1050,7516,1089],{"class":1088},[1050,7518,7281],{"class":2862},[1050,7520,2866],{"class":1088},[1050,7522,7523,7525,7527,7529,7531,7533,7535,7538,7540,7542,7544,7546,7548],{"class":1052,"line":1312},[1050,7524,7253],{"class":1067},[1050,7526,1089],{"class":1088},[1050,7528,7258],{"class":1126},[1050,7530,1130],{"class":1088},[1050,7532,7263],{"class":1148},[1050,7534,1436],{"class":1260},[1050,7536,7537],{"class":7268},"Kindle",[1050,7539,1436],{"class":1260},[1050,7541,1195],{"class":1088},[1050,7543,7276],{"class":1126},[1050,7545,1089],{"class":1088},[1050,7547,7281],{"class":2862},[1050,7549,2866],{"class":1088},[1050,7551,7552],{"class":1052,"line":1318},[1050,7553,2511],{"class":1088},[1050,7555,7556],{"class":1052,"line":1337},[1050,7557,1176],{"class":1067},[1050,7559,7560,7562,7564,7566,7568,7570,7572],{"class":1052,"line":1343},[1050,7561,1182],{"class":1148},[1050,7563,1282],{"class":1185},[1050,7565,1130],{"class":1088},[1050,7567,1192],{"class":1191},[1050,7569,1195],{"class":1088},[1050,7571,1291],{"class":1198},[1050,7573,1202],{"class":1088},[1050,7575,7576],{"class":1052,"line":1349},[1050,7577,7578],{"class":1056},"        # Detect device type\n",[1050,7580,7581,7584,7586,7588,7590,7592,7594,7596,7598,7600,7602,7604,7606,7608],{"class":1052,"line":1371},[1050,7582,7583],{"class":1067},"        user_agent ",[1050,7585,1118],{"class":1088},[1050,7587,1291],{"class":1067},[1050,7589,1089],{"class":1088},[1050,7591,2962],{"class":2862},[1050,7593,1089],{"class":1088},[1050,7595,2967],{"class":1126},[1050,7597,1130],{"class":1088},[1050,7599,1436],{"class":1260},[1050,7601,2974],{"class":1264},[1050,7603,1436],{"class":1260},[1050,7605,1195],{"class":1088},[1050,7607,2981],{"class":1260},[1050,7609,1137],{"class":1088},[1050,7611,7612],{"class":1052,"line":1376},[1050,7613,1340],{"class":1067},[1050,7615,7616,7618,7620,7623,7625,7627,7629,7631,7633,7635],{"class":1052,"line":1382},[1050,7617,1675],{"class":1067},[1050,7619,1089],{"class":1088},[1050,7621,7622],{"class":1231},"is_mobile",[1050,7624,1235],{"class":1088},[1050,7626,1357],{"class":1225},[1050,7628,1089],{"class":1088},[1050,7630,7622],{"class":1126},[1050,7632,1130],{"class":1088},[1050,7634,2951],{"class":1126},[1050,7636,1137],{"class":1088},[1050,7638,7639,7641,7643,7646,7648,7650,7652,7654,7656,7658],{"class":1052,"line":1398},[1050,7640,1675],{"class":1067},[1050,7642,1089],{"class":1088},[1050,7644,7645],{"class":1231},"is_tablet",[1050,7647,1235],{"class":1088},[1050,7649,1357],{"class":1225},[1050,7651,1089],{"class":1088},[1050,7653,7645],{"class":1126},[1050,7655,1130],{"class":1088},[1050,7657,2951],{"class":1126},[1050,7659,1137],{"class":1088},[1050,7661,7662,7664,7666,7669,7671,7673,7675,7677,7679,7681,7683,7685,7687,7689],{"class":1052,"line":1416},[1050,7663,1675],{"class":1067},[1050,7665,1089],{"class":1088},[1050,7667,7668],{"class":1231},"is_desktop",[1050,7670,1235],{"class":1088},[1050,7672,4219],{"class":1409},[1050,7674,5372],{"class":1088},[1050,7676,1366],{"class":1067},[1050,7678,1089],{"class":1088},[1050,7680,7622],{"class":1231},[1050,7682,5547],{"class":1409},[1050,7684,1291],{"class":1067},[1050,7686,1089],{"class":1088},[1050,7688,7645],{"class":1231},[1050,7690,1137],{"class":1088},[1050,7692,7693],{"class":1052,"line":1421},[1050,7694,1340],{"class":1067},[1050,7696,7697],{"class":1052,"line":1427},[1050,7698,7699],{"class":1056},"        # Add device info to request\n",[1050,7701,7702,7704,7706,7709,7711],{"class":1052,"line":1470},[1050,7703,1675],{"class":1067},[1050,7705,1089],{"class":1088},[1050,7707,7708],{"class":1231},"device_info",[1050,7710,1235],{"class":1088},[1050,7712,2773],{"class":1088},[1050,7714,7715,7717,7720,7722,7724,7726,7728,7731,7733,7735],{"class":1052,"line":1475},[1050,7716,2477],{"class":1260},[1050,7718,7719],{"class":1264},"type",[1050,7721,1436],{"class":1260},[1050,7723,2785],{"class":1088},[1050,7725,1357],{"class":1225},[1050,7727,1089],{"class":1088},[1050,7729,7730],{"class":1126},"get_device_type",[1050,7732,1130],{"class":1088},[1050,7734,1366],{"class":1126},[1050,7736,2866],{"class":1088},[1050,7738,7739,7741,7743,7745,7747,7750],{"class":1052,"line":1965},[1050,7740,2477],{"class":1260},[1050,7742,2951],{"class":1264},[1050,7744,1436],{"class":1260},[1050,7746,2785],{"class":1088},[1050,7748,7749],{"class":1067}," user_agent",[1050,7751,2485],{"class":1088},[1050,7753,7754,7756,7759,7761,7763,7765,7767,7770,7772,7774],{"class":1052,"line":1970},[1050,7755,2477],{"class":1260},[1050,7757,7758],{"class":1264},"screen_size",[1050,7760,1436],{"class":1260},[1050,7762,2785],{"class":1088},[1050,7764,1357],{"class":1225},[1050,7766,1089],{"class":1088},[1050,7768,7769],{"class":1126},"estimate_screen_size",[1050,7771,1130],{"class":1088},[1050,7773,1366],{"class":1126},[1050,7775,1137],{"class":1088},[1050,7777,7778],{"class":1052,"line":1992},[1050,7779,3085],{"class":1088},[1050,7781,7782],{"class":1052,"line":2002},[1050,7783,1340],{"class":1067},[1050,7785,7786,7788,7790,7792,7794,7796,7798,7800],{"class":1052,"line":2027},[1050,7787,1352],{"class":1067},[1050,7789,1118],{"class":1088},[1050,7791,1357],{"class":1225},[1050,7793,1089],{"class":1088},[1050,7795,1232],{"class":1126},[1050,7797,1130],{"class":1088},[1050,7799,1366],{"class":1126},[1050,7801,1137],{"class":1088},[1050,7803,7804],{"class":1052,"line":2072},[1050,7805,1340],{"class":1067},[1050,7807,7808],{"class":1052,"line":2077},[1050,7809,7810],{"class":1056},"        # Add device type header\n",[1050,7812,7813,7815,7817,7819,7822,7824,7826,7828,7830,7832,7834,7836,7838,7840,7842],{"class":1052,"line":2084},[1050,7814,1430],{"class":1067},[1050,7816,1433],{"class":1088},[1050,7818,1436],{"class":1260},[1050,7820,7821],{"class":1264},"X-Device-Type",[1050,7823,1436],{"class":1260},[1050,7825,1444],{"class":1088},[1050,7827,1235],{"class":1088},[1050,7829,1291],{"class":1067},[1050,7831,1089],{"class":1088},[1050,7833,7708],{"class":1231},[1050,7835,1433],{"class":1088},[1050,7837,1436],{"class":1260},[1050,7839,7719],{"class":1264},[1050,7841,1436],{"class":1260},[1050,7843,2335],{"class":1088},[1050,7845,7846],{"class":1052,"line":2089},[1050,7847,1340],{"class":1067},[1050,7849,7850,7852],{"class":1052,"line":2111},[1050,7851,1478],{"class":1063},[1050,7853,1481],{"class":1067},[1050,7855,7856],{"class":1052,"line":2121},[1050,7857,1176],{"class":1067},[1050,7859,7860,7862,7865,7867,7869,7871,7873],{"class":1052,"line":2127},[1050,7861,1182],{"class":1148},[1050,7863,7864],{"class":1644}," is_mobile",[1050,7866,1130],{"class":1088},[1050,7868,1192],{"class":1191},[1050,7870,1195],{"class":1088},[1050,7872,7749],{"class":1198},[1050,7874,1202],{"class":1088},[1050,7876,7877,7879,7882],{"class":1052,"line":2169},[1050,7878,1660],{"class":1162},[1050,7880,7881],{"class":1166},"Check if user agent indicates mobile device",[1050,7883,1170],{"class":1162},[1050,7885,7886,7888,7890,7892,7895,7897,7900,7902,7904,7906,7908,7911,7913,7915,7917,7919],{"class":1052,"line":2174},[1050,7887,1478],{"class":1063},[1050,7889,4033],{"class":1185},[1050,7891,1130],{"class":1088},[1050,7893,7894],{"class":1126},"pattern",[1050,7896,1089],{"class":1088},[1050,7898,7899],{"class":1126},"search",[1050,7901,1130],{"class":1088},[1050,7903,2951],{"class":1126},[1050,7905,1931],{"class":1088},[1050,7907,4054],{"class":1063},[1050,7909,7910],{"class":1126}," pattern ",[1050,7912,3501],{"class":1063},[1050,7914,1357],{"class":1225},[1050,7916,1089],{"class":1088},[1050,7918,7244],{"class":1231},[1050,7920,1137],{"class":1088},[1050,7922,7923],{"class":1052,"line":2180},[1050,7924,1176],{"class":1067},[1050,7926,7927,7929,7932,7934,7936,7938,7940],{"class":1052,"line":2201},[1050,7928,1182],{"class":1148},[1050,7930,7931],{"class":1644}," is_tablet",[1050,7933,1130],{"class":1088},[1050,7935,1192],{"class":1191},[1050,7937,1195],{"class":1088},[1050,7939,7749],{"class":1198},[1050,7941,1202],{"class":1088},[1050,7943,7944,7946,7949],{"class":1052,"line":2228},[1050,7945,1660],{"class":1162},[1050,7947,7948],{"class":1166},"Check if user agent indicates tablet device",[1050,7950,1170],{"class":1162},[1050,7952,7953,7955,7957,7959,7961,7963,7965,7967,7969,7971,7973,7975,7977,7979,7981,7983],{"class":1052,"line":2259},[1050,7954,1478],{"class":1063},[1050,7956,4033],{"class":1185},[1050,7958,1130],{"class":1088},[1050,7960,7894],{"class":1126},[1050,7962,1089],{"class":1088},[1050,7964,7899],{"class":1126},[1050,7966,1130],{"class":1088},[1050,7968,2951],{"class":1126},[1050,7970,1931],{"class":1088},[1050,7972,4054],{"class":1063},[1050,7974,7910],{"class":1126},[1050,7976,3501],{"class":1063},[1050,7978,1357],{"class":1225},[1050,7980,1089],{"class":1088},[1050,7982,7450],{"class":1231},[1050,7984,1137],{"class":1088},[1050,7986,7987],{"class":1052,"line":2264},[1050,7988,1176],{"class":1067},[1050,7990,7991,7993,7996,7998,8000,8002,8004],{"class":1052,"line":2271},[1050,7992,1182],{"class":1148},[1050,7994,7995],{"class":1644}," get_device_type",[1050,7997,1130],{"class":1088},[1050,7999,1192],{"class":1191},[1050,8001,1195],{"class":1088},[1050,8003,1291],{"class":1198},[1050,8005,1202],{"class":1088},[1050,8007,8008,8010,8013],{"class":1052,"line":2276},[1050,8009,1660],{"class":1162},[1050,8011,8012],{"class":1166},"Get device type string",[1050,8014,1170],{"class":1162},[1050,8016,8017,8019,8021,8023,8025],{"class":1052,"line":2290},[1050,8018,2005],{"class":1063},[1050,8020,1291],{"class":1067},[1050,8022,1089],{"class":1088},[1050,8024,7645],{"class":1231},[1050,8026,1156],{"class":1088},[1050,8028,8029,8031,8033,8036],{"class":1052,"line":2300},[1050,8030,4229],{"class":1063},[1050,8032,1915],{"class":1260},[1050,8034,8035],{"class":1264},"tablet",[1050,8037,2506],{"class":1260},[1050,8039,8040,8042,8044,8046,8048],{"class":1052,"line":2309},[1050,8041,6955],{"class":1063},[1050,8043,1291],{"class":1067},[1050,8045,1089],{"class":1088},[1050,8047,7622],{"class":1231},[1050,8049,1156],{"class":1088},[1050,8051,8052,8054,8056,8059],{"class":1052,"line":3319},[1050,8053,4229],{"class":1063},[1050,8055,1915],{"class":1260},[1050,8057,8058],{"class":1264},"mobile",[1050,8060,2506],{"class":1260},[1050,8062,8063,8065],{"class":1052,"line":3329},[1050,8064,3402],{"class":1063},[1050,8066,1156],{"class":1088},[1050,8068,8069,8071,8073,8076],{"class":1052,"line":3358},[1050,8070,4229],{"class":1063},[1050,8072,1915],{"class":1260},[1050,8074,8075],{"class":1264},"desktop",[1050,8077,2506],{"class":1260},[1050,8079,8080],{"class":1052,"line":3368},[1050,8081,1176],{"class":1067},[1050,8083,8084,8086,8089,8091,8093,8095,8097],{"class":1052,"line":3399},[1050,8085,1182],{"class":1148},[1050,8087,8088],{"class":1644}," estimate_screen_size",[1050,8090,1130],{"class":1088},[1050,8092,1192],{"class":1191},[1050,8094,1195],{"class":1088},[1050,8096,1291],{"class":1198},[1050,8098,1202],{"class":1088},[1050,8100,8101,8103,8106],{"class":1052,"line":3407},[1050,8102,1660],{"class":1162},[1050,8104,8105],{"class":1166},"Estimate screen size category",[1050,8107,1170],{"class":1162},[1050,8109,8110,8112,8114,8116,8118],{"class":1052,"line":3435},[1050,8111,2005],{"class":1063},[1050,8113,1291],{"class":1067},[1050,8115,1089],{"class":1088},[1050,8117,7622],{"class":1231},[1050,8119,1156],{"class":1088},[1050,8121,8122,8124,8126,8129],{"class":1052,"line":3443},[1050,8123,4229],{"class":1063},[1050,8125,1915],{"class":1260},[1050,8127,8128],{"class":1264},"small",[1050,8130,2506],{"class":1260},[1050,8132,8133,8135,8137,8139,8141],{"class":1052,"line":3448},[1050,8134,6955],{"class":1063},[1050,8136,1291],{"class":1067},[1050,8138,1089],{"class":1088},[1050,8140,7645],{"class":1231},[1050,8142,1156],{"class":1088},[1050,8144,8145,8147,8149,8152],{"class":1052,"line":3466},[1050,8146,4229],{"class":1063},[1050,8148,1915],{"class":1260},[1050,8150,8151],{"class":1264},"medium",[1050,8153,2506],{"class":1260},[1050,8155,8156,8158],{"class":1052,"line":3476},[1050,8157,3402],{"class":1063},[1050,8159,1156],{"class":1088},[1050,8161,8162,8164,8166,8169],{"class":1052,"line":3487},[1050,8163,4229],{"class":1063},[1050,8165,1915],{"class":1260},[1050,8167,8168],{"class":1264},"large",[1050,8170,2506],{"class":1260},[1030,8172,8174],{"id":8173},"advanced-middleware-patterns","Advanced Middleware Patterns",[1035,8176,8178],{"id":8177},"conditional-middleware","Conditional Middleware",[1040,8180,8182],{"className":1042,"code":8181,"language":1044,"meta":1045,"style":1045},"class ConditionalMiddleware:\n    \"\"\"Middleware that runs conditionally based on settings\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n        \n        # Load configuration\n        from django.conf import settings\n        self.config = getattr(settings, 'CONDITIONAL_MIDDLEWARE_CONFIG', {})\n        \n        self.enabled = self.config.get('enabled', True)\n        self.debug_mode = self.config.get('debug', False)\n        self.allowed_ips = self.config.get('allowed_ips', [])\n        self.excluded_paths = self.config.get('excluded_paths', [])\n    \n    def __call__(self, request):\n        # Check if middleware should run\n        if not self.should_process(request):\n            return self.get_response(request)\n        \n        # Process request\n        if self.debug_mode:\n            print(f\"ConditionalMiddleware processing: {request.path}\")\n        \n        response = self.get_response(request)\n        \n        # Process response\n        response['X-Processed-By'] = 'ConditionalMiddleware'\n        \n        return response\n    \n    def should_process(self, request):\n        \"\"\"Determine if middleware should process this request\"\"\"\n        if not self.enabled:\n            return False\n        \n        # Check IP whitelist\n        if self.allowed_ips:\n            client_ip = self.get_client_ip(request)\n            if client_ip not in self.allowed_ips:\n                return False\n        \n        # Check excluded paths\n        if any(request.path.startswith(path) for path in self.excluded_paths):\n            return False\n        \n        return True\n    \n    def get_client_ip(self, request):\n        \"\"\"Get client IP address\"\"\"\n        x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')\n        if x_forwarded_for:\n            ip = x_forwarded_for.split(',')[0]\n        else:\n            ip = request.META.get('REMOTE_ADDR')\n        return ip\n",[1047,8183,8184,8193,8202,8206,8222,8234,8238,8243,8260,8291,8295,8331,8368,8402,8434,8438,8454,8459,8478,8494,8498,8502,8514,8540,8544,8562,8566,8570,8594,8598,8604,8608,8625,8634,8648,8654,8658,8663,8675,8694,8713,8719,8723,8728,8766,8772,8776,8782,8786,8802,8810,8836,8844,8870,8876,8902],{"__ignoreMap":1045},[1050,8185,8186,8188,8191],{"class":1052,"line":1053},[1050,8187,1149],{"class":1148},[1050,8189,8190],{"class":1152}," ConditionalMiddleware",[1050,8192,1156],{"class":1088},[1050,8194,8195,8197,8200],{"class":1052,"line":1060},[1050,8196,1163],{"class":1162},[1050,8198,8199],{"class":1166},"Middleware that runs conditionally based on settings",[1050,8201,1170],{"class":1162},[1050,8203,8204],{"class":1052,"line":1071},[1050,8205,1176],{"class":1067},[1050,8207,8208,8210,8212,8214,8216,8218,8220],{"class":1052,"line":1079},[1050,8209,1182],{"class":1148},[1050,8211,1186],{"class":1185},[1050,8213,1130],{"class":1088},[1050,8215,1192],{"class":1191},[1050,8217,1195],{"class":1088},[1050,8219,1199],{"class":1198},[1050,8221,1202],{"class":1088},[1050,8223,8224,8226,8228,8230,8232],{"class":1052,"line":1105},[1050,8225,1226],{"class":1225},[1050,8227,1089],{"class":1088},[1050,8229,1232],{"class":1231},[1050,8231,1235],{"class":1088},[1050,8233,1238],{"class":1067},[1050,8235,8236],{"class":1052,"line":1112},[1050,8237,1340],{"class":1067},[1050,8239,8240],{"class":1052,"line":1140},[1050,8241,8242],{"class":1056},"        # Load configuration\n",[1050,8244,8245,8248,8250,8252,8255,8257],{"class":1052,"line":1145},[1050,8246,8247],{"class":1063},"        from",[1050,8249,1085],{"class":1067},[1050,8251,1089],{"class":1088},[1050,8253,8254],{"class":1067},"conf ",[1050,8256,1064],{"class":1063},[1050,8258,8259],{"class":1067}," settings\n",[1050,8261,8262,8264,8266,8268,8270,8272,8274,8277,8279,8281,8284,8286,8288],{"class":1052,"line":1159},[1050,8263,1226],{"class":1225},[1050,8265,1089],{"class":1088},[1050,8267,4535],{"class":1231},[1050,8269,1235],{"class":1088},[1050,8271,2050],{"class":1185},[1050,8273,1130],{"class":1088},[1050,8275,8276],{"class":1126},"settings",[1050,8278,1195],{"class":1088},[1050,8280,1915],{"class":1260},[1050,8282,8283],{"class":1264},"CONDITIONAL_MIDDLEWARE_CONFIG",[1050,8285,1436],{"class":1260},[1050,8287,1195],{"class":1088},[1050,8289,8290],{"class":1088}," {})\n",[1050,8292,8293],{"class":1052,"line":1173},[1050,8294,1340],{"class":1067},[1050,8296,8297,8299,8301,8304,8306,8308,8310,8312,8314,8316,8318,8320,8322,8324,8326,8329],{"class":1052,"line":1179},[1050,8298,1226],{"class":1225},[1050,8300,1089],{"class":1088},[1050,8302,8303],{"class":1231},"enabled",[1050,8305,1235],{"class":1088},[1050,8307,1357],{"class":1225},[1050,8309,1089],{"class":1088},[1050,8311,4535],{"class":1231},[1050,8313,1089],{"class":1088},[1050,8315,2967],{"class":1126},[1050,8317,1130],{"class":1088},[1050,8319,1436],{"class":1260},[1050,8321,8303],{"class":1264},[1050,8323,1436],{"class":1260},[1050,8325,1195],{"class":1088},[1050,8327,8328],{"class":1849}," True",[1050,8330,1137],{"class":1088},[1050,8332,8333,8335,8337,8340,8342,8344,8346,8348,8350,8352,8354,8356,8359,8361,8363,8366],{"class":1052,"line":1205},[1050,8334,1226],{"class":1225},[1050,8336,1089],{"class":1088},[1050,8338,8339],{"class":1231},"debug_mode",[1050,8341,1235],{"class":1088},[1050,8343,1357],{"class":1225},[1050,8345,1089],{"class":1088},[1050,8347,4535],{"class":1231},[1050,8349,1089],{"class":1088},[1050,8351,2967],{"class":1126},[1050,8353,1130],{"class":1088},[1050,8355,1436],{"class":1260},[1050,8357,8358],{"class":1264},"debug",[1050,8360,1436],{"class":1260},[1050,8362,1195],{"class":1088},[1050,8364,8365],{"class":1849}," False",[1050,8367,1137],{"class":1088},[1050,8369,8370,8372,8374,8377,8379,8381,8383,8385,8387,8389,8391,8393,8395,8397,8399],{"class":1052,"line":1211},[1050,8371,1226],{"class":1225},[1050,8373,1089],{"class":1088},[1050,8375,8376],{"class":1231},"allowed_ips",[1050,8378,1235],{"class":1088},[1050,8380,1357],{"class":1225},[1050,8382,1089],{"class":1088},[1050,8384,4535],{"class":1231},[1050,8386,1089],{"class":1088},[1050,8388,2967],{"class":1126},[1050,8390,1130],{"class":1088},[1050,8392,1436],{"class":1260},[1050,8394,8376],{"class":1264},[1050,8396,1436],{"class":1260},[1050,8398,1195],{"class":1088},[1050,8400,8401],{"class":1088}," [])\n",[1050,8403,8404,8406,8408,8410,8412,8414,8416,8418,8420,8422,8424,8426,8428,8430,8432],{"class":1052,"line":1217},[1050,8405,1226],{"class":1225},[1050,8407,1089],{"class":1088},[1050,8409,5287],{"class":1231},[1050,8411,1235],{"class":1088},[1050,8413,1357],{"class":1225},[1050,8415,1089],{"class":1088},[1050,8417,4535],{"class":1231},[1050,8419,1089],{"class":1088},[1050,8421,2967],{"class":1126},[1050,8423,1130],{"class":1088},[1050,8425,1436],{"class":1260},[1050,8427,5287],{"class":1264},[1050,8429,1436],{"class":1260},[1050,8431,1195],{"class":1088},[1050,8433,8401],{"class":1088},[1050,8435,8436],{"class":1052,"line":1222},[1050,8437,1176],{"class":1067},[1050,8439,8440,8442,8444,8446,8448,8450,8452],{"class":1052,"line":1241},[1050,8441,1182],{"class":1148},[1050,8443,1282],{"class":1185},[1050,8445,1130],{"class":1088},[1050,8447,1192],{"class":1191},[1050,8449,1195],{"class":1088},[1050,8451,1291],{"class":1198},[1050,8453,1202],{"class":1088},[1050,8455,8456],{"class":1052,"line":1247},[1050,8457,8458],{"class":1056},"        # Check if middleware should run\n",[1050,8460,8461,8463,8465,8467,8469,8472,8474,8476],{"class":1052,"line":1272},[1050,8462,2005],{"class":1063},[1050,8464,4219],{"class":1409},[1050,8466,1357],{"class":1225},[1050,8468,1089],{"class":1088},[1050,8470,8471],{"class":1126},"should_process",[1050,8473,1130],{"class":1088},[1050,8475,1366],{"class":1126},[1050,8477,1202],{"class":1088},[1050,8479,8480,8482,8484,8486,8488,8490,8492],{"class":1052,"line":1277},[1050,8481,4229],{"class":1063},[1050,8483,1357],{"class":1225},[1050,8485,1089],{"class":1088},[1050,8487,1232],{"class":1126},[1050,8489,1130],{"class":1088},[1050,8491,1366],{"class":1126},[1050,8493,1137],{"class":1088},[1050,8495,8496],{"class":1052,"line":1296},[1050,8497,1340],{"class":1067},[1050,8499,8500],{"class":1052,"line":1301},[1050,8501,1315],{"class":1056},[1050,8503,8504,8506,8508,8510,8512],{"class":1052,"line":1307},[1050,8505,2005],{"class":1063},[1050,8507,1357],{"class":1225},[1050,8509,1089],{"class":1088},[1050,8511,8339],{"class":1231},[1050,8513,1156],{"class":1088},[1050,8515,8516,8519,8521,8523,8526,8528,8530,8532,8534,8536,8538],{"class":1052,"line":1312},[1050,8517,8518],{"class":1185},"            print",[1050,8520,1130],{"class":1088},[1050,8522,1727],{"class":1148},[1050,8524,8525],{"class":1264},"\"ConditionalMiddleware processing: ",[1050,8527,1455],{"class":1454},[1050,8529,1366],{"class":1126},[1050,8531,1089],{"class":1088},[1050,8533,1752],{"class":1231},[1050,8535,1464],{"class":1454},[1050,8537,1261],{"class":1264},[1050,8539,1137],{"class":1088},[1050,8541,8542],{"class":1052,"line":1318},[1050,8543,1340],{"class":1067},[1050,8545,8546,8548,8550,8552,8554,8556,8558,8560],{"class":1052,"line":1337},[1050,8547,1352],{"class":1067},[1050,8549,1118],{"class":1088},[1050,8551,1357],{"class":1225},[1050,8553,1089],{"class":1088},[1050,8555,1232],{"class":1126},[1050,8557,1130],{"class":1088},[1050,8559,1366],{"class":1126},[1050,8561,1137],{"class":1088},[1050,8563,8564],{"class":1052,"line":1343},[1050,8565,1340],{"class":1067},[1050,8567,8568],{"class":1052,"line":1349},[1050,8569,1379],{"class":1056},[1050,8571,8572,8574,8576,8578,8581,8583,8585,8587,8589,8592],{"class":1052,"line":1371},[1050,8573,1430],{"class":1067},[1050,8575,1433],{"class":1088},[1050,8577,1436],{"class":1260},[1050,8579,8580],{"class":1264},"X-Processed-By",[1050,8582,1436],{"class":1260},[1050,8584,1444],{"class":1088},[1050,8586,1235],{"class":1088},[1050,8588,1915],{"class":1260},[1050,8590,8591],{"class":1264},"ConditionalMiddleware",[1050,8593,2506],{"class":1260},[1050,8595,8596],{"class":1052,"line":1376},[1050,8597,1340],{"class":1067},[1050,8599,8600,8602],{"class":1052,"line":1382},[1050,8601,1478],{"class":1063},[1050,8603,1481],{"class":1067},[1050,8605,8606],{"class":1052,"line":1398},[1050,8607,1176],{"class":1067},[1050,8609,8610,8612,8615,8617,8619,8621,8623],{"class":1052,"line":1416},[1050,8611,1182],{"class":1148},[1050,8613,8614],{"class":1644}," should_process",[1050,8616,1130],{"class":1088},[1050,8618,1192],{"class":1191},[1050,8620,1195],{"class":1088},[1050,8622,1291],{"class":1198},[1050,8624,1202],{"class":1088},[1050,8626,8627,8629,8632],{"class":1052,"line":1421},[1050,8628,1660],{"class":1162},[1050,8630,8631],{"class":1166},"Determine if middleware should process this request",[1050,8633,1170],{"class":1162},[1050,8635,8636,8638,8640,8642,8644,8646],{"class":1052,"line":1427},[1050,8637,2005],{"class":1063},[1050,8639,4219],{"class":1409},[1050,8641,1357],{"class":1225},[1050,8643,1089],{"class":1088},[1050,8645,8303],{"class":1231},[1050,8647,1156],{"class":1088},[1050,8649,8650,8652],{"class":1052,"line":1470},[1050,8651,4229],{"class":1063},[1050,8653,4232],{"class":1849},[1050,8655,8656],{"class":1052,"line":1475},[1050,8657,1340],{"class":1067},[1050,8659,8660],{"class":1052,"line":1965},[1050,8661,8662],{"class":1056},"        # Check IP whitelist\n",[1050,8664,8665,8667,8669,8671,8673],{"class":1052,"line":1970},[1050,8666,2005],{"class":1063},[1050,8668,1357],{"class":1225},[1050,8670,1089],{"class":1088},[1050,8672,8376],{"class":1231},[1050,8674,1156],{"class":1088},[1050,8676,8677,8680,8682,8684,8686,8688,8690,8692],{"class":1052,"line":1992},[1050,8678,8679],{"class":1067},"            client_ip ",[1050,8681,1118],{"class":1088},[1050,8683,1357],{"class":1225},[1050,8685,1089],{"class":1088},[1050,8687,2938],{"class":1126},[1050,8689,1130],{"class":1088},[1050,8691,1366],{"class":1126},[1050,8693,1137],{"class":1088},[1050,8695,8696,8698,8701,8703,8705,8707,8709,8711],{"class":1052,"line":2002},[1050,8697,3521],{"class":1063},[1050,8699,8700],{"class":1067}," client_ip ",[1050,8702,3548],{"class":1409},[1050,8704,3551],{"class":1409},[1050,8706,1357],{"class":1225},[1050,8708,1089],{"class":1088},[1050,8710,8376],{"class":1231},[1050,8712,1156],{"class":1088},[1050,8714,8715,8717],{"class":1052,"line":2027},[1050,8716,3906],{"class":1063},[1050,8718,4232],{"class":1849},[1050,8720,8721],{"class":1052,"line":2072},[1050,8722,1340],{"class":1067},[1050,8724,8725],{"class":1052,"line":2077},[1050,8726,8727],{"class":1056},"        # Check excluded paths\n",[1050,8729,8730,8732,8734,8736,8738,8740,8742,8744,8746,8748,8750,8752,8754,8756,8758,8760,8762,8764],{"class":1052,"line":2084},[1050,8731,2005],{"class":1063},[1050,8733,4033],{"class":1185},[1050,8735,1130],{"class":1088},[1050,8737,1366],{"class":1126},[1050,8739,1089],{"class":1088},[1050,8741,1752],{"class":1231},[1050,8743,1089],{"class":1088},[1050,8745,3528],{"class":1126},[1050,8747,1130],{"class":1088},[1050,8749,1752],{"class":1126},[1050,8751,1931],{"class":1088},[1050,8753,4054],{"class":1063},[1050,8755,4057],{"class":1126},[1050,8757,3501],{"class":1063},[1050,8759,1357],{"class":1225},[1050,8761,1089],{"class":1088},[1050,8763,5287],{"class":1231},[1050,8765,1202],{"class":1088},[1050,8767,8768,8770],{"class":1052,"line":2089},[1050,8769,4229],{"class":1063},[1050,8771,4232],{"class":1849},[1050,8773,8774],{"class":1052,"line":2111},[1050,8775,1340],{"class":1067},[1050,8777,8778,8780],{"class":1052,"line":2121},[1050,8779,1478],{"class":1063},[1050,8781,4381],{"class":1849},[1050,8783,8784],{"class":1052,"line":2127},[1050,8785,1176],{"class":1067},[1050,8787,8788,8790,8792,8794,8796,8798,8800],{"class":1052,"line":2169},[1050,8789,1182],{"class":1148},[1050,8791,3306],{"class":1644},[1050,8793,1130],{"class":1088},[1050,8795,1192],{"class":1191},[1050,8797,1195],{"class":1088},[1050,8799,1291],{"class":1198},[1050,8801,1202],{"class":1088},[1050,8803,8804,8806,8808],{"class":1052,"line":2174},[1050,8805,1660],{"class":1162},[1050,8807,5015],{"class":1166},[1050,8809,1170],{"class":1162},[1050,8811,8812,8814,8816,8818,8820,8822,8824,8826,8828,8830,8832,8834],{"class":1052,"line":2180},[1050,8813,3332],{"class":1067},[1050,8815,1118],{"class":1088},[1050,8817,1291],{"class":1067},[1050,8819,1089],{"class":1088},[1050,8821,2962],{"class":2862},[1050,8823,1089],{"class":1088},[1050,8825,2967],{"class":1126},[1050,8827,1130],{"class":1088},[1050,8829,1436],{"class":1260},[1050,8831,3351],{"class":1264},[1050,8833,1436],{"class":1260},[1050,8835,1137],{"class":1088},[1050,8837,8838,8840,8842],{"class":1052,"line":2201},[1050,8839,2005],{"class":1063},[1050,8841,3363],{"class":1067},[1050,8843,1156],{"class":1088},[1050,8845,8846,8848,8850,8852,8854,8856,8858,8860,8862,8864,8866,8868],{"class":1052,"line":2228},[1050,8847,3371],{"class":1067},[1050,8849,1118],{"class":1088},[1050,8851,3363],{"class":1067},[1050,8853,1089],{"class":1088},[1050,8855,3380],{"class":1126},[1050,8857,1130],{"class":1088},[1050,8859,1436],{"class":1260},[1050,8861,1195],{"class":1264},[1050,8863,1436],{"class":1260},[1050,8865,3391],{"class":1088},[1050,8867,3394],{"class":2331},[1050,8869,2335],{"class":1088},[1050,8871,8872,8874],{"class":1052,"line":2259},[1050,8873,3402],{"class":1063},[1050,8875,1156],{"class":1088},[1050,8877,8878,8880,8882,8884,8886,8888,8890,8892,8894,8896,8898,8900],{"class":1052,"line":2264},[1050,8879,3371],{"class":1067},[1050,8881,1118],{"class":1088},[1050,8883,1291],{"class":1067},[1050,8885,1089],{"class":1088},[1050,8887,2962],{"class":2862},[1050,8889,1089],{"class":1088},[1050,8891,2967],{"class":1126},[1050,8893,1130],{"class":1088},[1050,8895,1436],{"class":1260},[1050,8897,3428],{"class":1264},[1050,8899,1436],{"class":1260},[1050,8901,1137],{"class":1088},[1050,8903,8904,8906],{"class":1052,"line":2271},[1050,8905,1478],{"class":1063},[1050,8907,3440],{"class":1067},[1035,8909,8911],{"id":8910},"middleware-with-database-integration","Middleware with Database Integration",[1040,8913,8915],{"className":1042,"code":8914,"language":1044,"meta":1045,"style":1045},"from .models import RequestLog, BlockedIP\n\nclass DatabaseIntegratedMiddleware:\n    \"\"\"Middleware that integrates with database models\"\"\"\n    \n    def __init__(self, get_response):\n        self.get_response = get_response\n    \n    def __call__(self, request):\n        # Check if IP is blocked\n        client_ip = self.get_client_ip(request)\n        if self.is_ip_blocked(client_ip):\n            return HttpResponseForbidden('IP address blocked')\n        \n        # Log request to database\n        request_log = self.create_request_log(request)\n        \n        response = self.get_response(request)\n        \n        # Update request log with response info\n        self.update_request_log(request_log, response)\n        \n        return response\n    \n    def is_ip_blocked(self, ip_address):\n        \"\"\"Check if IP address is blocked\"\"\"\n        try:\n            return BlockedIP.objects.filter(\n                ip_address=ip_address,\n                is_active=True\n            ).exists()\n        except Exception:\n            return False\n    \n    def create_request_log(self, request):\n        \"\"\"Create request log entry\"\"\"\n        try:\n            return RequestLog.objects.create(\n                ip_address=self.get_client_ip(request),\n                path=request.path,\n                method=request.method,\n                user_agent=request.META.get('HTTP_USER_AGENT', ''),\n                user=request.user if hasattr(request, 'user') and request.user.is_authenticated else None,\n                timestamp=timezone.now()\n            )\n        except Exception as e:\n            logger.error(f\"Error creating request log: {e}\")\n            return None\n    \n    def update_request_log(self, request_log, response):\n        \"\"\"Update request log with response information\"\"\"\n        if request_log:\n            try:\n                request_log.status_code = response.status_code\n                request_log.response_size = len(response.content) if hasattr(response, 'content') else 0\n                request_log.save(update_fields=['status_code', 'response_size'])\n            except Exception as e:\n                logger.error(f\"Error updating request log: {e}\")\n    \n    def get_client_ip(self, request):\n        \"\"\"Get client IP address\"\"\"\n        x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')\n        if x_forwarded_for:\n            ip = x_forwarded_for.split(',')[0]\n        else:\n            ip = request.META.get('REMOTE_ADDR')\n        return ip\n\n# models.py\nclass RequestLog(models.Model):\n    ip_address = models.GenericIPAddressField()\n    path = models.CharField(max_length=255)\n    method = models.CharField(max_length=10)\n    user_agent = models.TextField()\n    user = models.ForeignKey(User, on_delete=models.SET_NULL, null=True, blank=True)\n    timestamp = models.DateTimeField()\n    status_code = models.IntegerField(null=True, blank=True)\n    response_size = models.IntegerField(null=True, blank=True)\n\nclass BlockedIP(models.Model):\n    ip_address = models.GenericIPAddressField(unique=True)\n    reason = models.CharField(max_length=255)\n    blocked_at = models.DateTimeField(auto_now_add=True)\n    is_active = models.BooleanField(default=True)\n",[1047,8916,8917,8935,8939,8948,8957,8961,8977,8989,8993,9009,9014,9032,9049,9067,9071,9076,9096,9100,9118,9122,9127,9147,9151,9157,9161,9179,9188,9194,9212,9223,9233,9243,9251,9257,9261,9278,9287,9293,9309,9327,9342,9357,9388,9438,9454,9459,9471,9496,9502,9506,9528,9537,9545,9552,9569,9615,9645,9658,9684,9688,9704,9712,9738,9746,9772,9778,9804,9810,9814,9818,9834,9848,9870,9892,9906,9956,9970,10001,10032,10036,10052,10075,10098,10122],{"__ignoreMap":1045},[1050,8918,8919,8921,8923,8925,8927,8930,8932],{"class":1052,"line":1053},[1050,8920,1082],{"class":1063},[1050,8922,5177],{"class":1088},[1050,8924,5151],{"class":1067},[1050,8926,1064],{"class":1063},[1050,8928,8929],{"class":1067}," RequestLog",[1050,8931,1195],{"class":1088},[1050,8933,8934],{"class":1067}," BlockedIP\n",[1050,8936,8937],{"class":1052,"line":1060},[1050,8938,1109],{"emptyLinePlaceholder":1108},[1050,8940,8941,8943,8946],{"class":1052,"line":1071},[1050,8942,1149],{"class":1148},[1050,8944,8945],{"class":1152}," DatabaseIntegratedMiddleware",[1050,8947,1156],{"class":1088},[1050,8949,8950,8952,8955],{"class":1052,"line":1079},[1050,8951,1163],{"class":1162},[1050,8953,8954],{"class":1166},"Middleware that integrates with database models",[1050,8956,1170],{"class":1162},[1050,8958,8959],{"class":1052,"line":1105},[1050,8960,1176],{"class":1067},[1050,8962,8963,8965,8967,8969,8971,8973,8975],{"class":1052,"line":1112},[1050,8964,1182],{"class":1148},[1050,8966,1186],{"class":1185},[1050,8968,1130],{"class":1088},[1050,8970,1192],{"class":1191},[1050,8972,1195],{"class":1088},[1050,8974,1199],{"class":1198},[1050,8976,1202],{"class":1088},[1050,8978,8979,8981,8983,8985,8987],{"class":1052,"line":1140},[1050,8980,1226],{"class":1225},[1050,8982,1089],{"class":1088},[1050,8984,1232],{"class":1231},[1050,8986,1235],{"class":1088},[1050,8988,1238],{"class":1067},[1050,8990,8991],{"class":1052,"line":1145},[1050,8992,1176],{"class":1067},[1050,8994,8995,8997,8999,9001,9003,9005,9007],{"class":1052,"line":1159},[1050,8996,1182],{"class":1148},[1050,8998,1282],{"class":1185},[1050,9000,1130],{"class":1088},[1050,9002,1192],{"class":1191},[1050,9004,1195],{"class":1088},[1050,9006,1291],{"class":1198},[1050,9008,1202],{"class":1088},[1050,9010,9011],{"class":1052,"line":1173},[1050,9012,9013],{"class":1056},"        # Check if IP is blocked\n",[1050,9015,9016,9018,9020,9022,9024,9026,9028,9030],{"class":1052,"line":1179},[1050,9017,4246],{"class":1067},[1050,9019,1118],{"class":1088},[1050,9021,1357],{"class":1225},[1050,9023,1089],{"class":1088},[1050,9025,2938],{"class":1126},[1050,9027,1130],{"class":1088},[1050,9029,1366],{"class":1126},[1050,9031,1137],{"class":1088},[1050,9033,9034,9036,9038,9040,9043,9045,9047],{"class":1052,"line":1205},[1050,9035,2005],{"class":1063},[1050,9037,1357],{"class":1225},[1050,9039,1089],{"class":1088},[1050,9041,9042],{"class":1126},"is_ip_blocked",[1050,9044,1130],{"class":1088},[1050,9046,4277],{"class":1126},[1050,9048,1202],{"class":1088},[1050,9050,9051,9053,9056,9058,9060,9063,9065],{"class":1052,"line":1211},[1050,9052,4229],{"class":1063},[1050,9054,9055],{"class":1126}," HttpResponseForbidden",[1050,9057,1130],{"class":1088},[1050,9059,1436],{"class":1260},[1050,9061,9062],{"class":1264},"IP address blocked",[1050,9064,1436],{"class":1260},[1050,9066,1137],{"class":1088},[1050,9068,9069],{"class":1052,"line":1217},[1050,9070,1340],{"class":1067},[1050,9072,9073],{"class":1052,"line":1222},[1050,9074,9075],{"class":1056},"        # Log request to database\n",[1050,9077,9078,9081,9083,9085,9087,9090,9092,9094],{"class":1052,"line":1241},[1050,9079,9080],{"class":1067},"        request_log ",[1050,9082,1118],{"class":1088},[1050,9084,1357],{"class":1225},[1050,9086,1089],{"class":1088},[1050,9088,9089],{"class":1126},"create_request_log",[1050,9091,1130],{"class":1088},[1050,9093,1366],{"class":1126},[1050,9095,1137],{"class":1088},[1050,9097,9098],{"class":1052,"line":1247},[1050,9099,1340],{"class":1067},[1050,9101,9102,9104,9106,9108,9110,9112,9114,9116],{"class":1052,"line":1272},[1050,9103,1352],{"class":1067},[1050,9105,1118],{"class":1088},[1050,9107,1357],{"class":1225},[1050,9109,1089],{"class":1088},[1050,9111,1232],{"class":1126},[1050,9113,1130],{"class":1088},[1050,9115,1366],{"class":1126},[1050,9117,1137],{"class":1088},[1050,9119,9120],{"class":1052,"line":1277},[1050,9121,1340],{"class":1067},[1050,9123,9124],{"class":1052,"line":1296},[1050,9125,9126],{"class":1056},"        # Update request log with response info\n",[1050,9128,9129,9131,9133,9136,9138,9141,9143,9145],{"class":1052,"line":1301},[1050,9130,1226],{"class":1225},[1050,9132,1089],{"class":1088},[1050,9134,9135],{"class":1126},"update_request_log",[1050,9137,1130],{"class":1088},[1050,9139,9140],{"class":1126},"request_log",[1050,9142,1195],{"class":1088},[1050,9144,1631],{"class":1126},[1050,9146,1137],{"class":1088},[1050,9148,9149],{"class":1052,"line":1307},[1050,9150,1340],{"class":1067},[1050,9152,9153,9155],{"class":1052,"line":1312},[1050,9154,1478],{"class":1063},[1050,9156,1481],{"class":1067},[1050,9158,9159],{"class":1052,"line":1318},[1050,9160,1176],{"class":1067},[1050,9162,9163,9165,9168,9170,9172,9174,9177],{"class":1052,"line":1337},[1050,9164,1182],{"class":1148},[1050,9166,9167],{"class":1644}," is_ip_blocked",[1050,9169,1130],{"class":1088},[1050,9171,1192],{"class":1191},[1050,9173,1195],{"class":1088},[1050,9175,9176],{"class":1198}," ip_address",[1050,9178,1202],{"class":1088},[1050,9180,9181,9183,9186],{"class":1052,"line":1343},[1050,9182,1660],{"class":1162},[1050,9184,9185],{"class":1166},"Check if IP address is blocked",[1050,9187,1170],{"class":1162},[1050,9189,9190,9192],{"class":1052,"line":1349},[1050,9191,5679],{"class":1063},[1050,9193,1156],{"class":1088},[1050,9195,9196,9198,9201,9203,9205,9207,9210],{"class":1052,"line":1371},[1050,9197,4229],{"class":1063},[1050,9199,9200],{"class":1067}," BlockedIP",[1050,9202,1089],{"class":1088},[1050,9204,5861],{"class":1231},[1050,9206,1089],{"class":1088},[1050,9208,9209],{"class":1126},"filter",[1050,9211,4614],{"class":1088},[1050,9213,9214,9217,9219,9221],{"class":1052,"line":1376},[1050,9215,9216],{"class":4634},"                ip_address",[1050,9218,1118],{"class":1088},[1050,9220,2927],{"class":1126},[1050,9222,2485],{"class":1088},[1050,9224,9225,9228,9230],{"class":1052,"line":1382},[1050,9226,9227],{"class":4634},"                is_active",[1050,9229,1118],{"class":1088},[1050,9231,9232],{"class":1849},"True\n",[1050,9234,9235,9238,9241],{"class":1052,"line":1398},[1050,9236,9237],{"class":1088},"            ).",[1050,9239,9240],{"class":1126},"exists",[1050,9242,1334],{"class":1088},[1050,9244,9245,9247,9249],{"class":1052,"line":1416},[1050,9246,5947],{"class":1063},[1050,9248,5950],{"class":2314},[1050,9250,1156],{"class":1088},[1050,9252,9253,9255],{"class":1052,"line":1421},[1050,9254,4229],{"class":1063},[1050,9256,4232],{"class":1849},[1050,9258,9259],{"class":1052,"line":1427},[1050,9260,1176],{"class":1067},[1050,9262,9263,9265,9268,9270,9272,9274,9276],{"class":1052,"line":1470},[1050,9264,1182],{"class":1148},[1050,9266,9267],{"class":1644}," create_request_log",[1050,9269,1130],{"class":1088},[1050,9271,1192],{"class":1191},[1050,9273,1195],{"class":1088},[1050,9275,1291],{"class":1198},[1050,9277,1202],{"class":1088},[1050,9279,9280,9282,9285],{"class":1052,"line":1475},[1050,9281,1660],{"class":1162},[1050,9283,9284],{"class":1166},"Create request log entry",[1050,9286,1170],{"class":1162},[1050,9288,9289,9291],{"class":1052,"line":1965},[1050,9290,5679],{"class":1063},[1050,9292,1156],{"class":1088},[1050,9294,9295,9297,9299,9301,9303,9305,9307],{"class":1052,"line":1970},[1050,9296,4229],{"class":1063},[1050,9298,8929],{"class":1067},[1050,9300,1089],{"class":1088},[1050,9302,5861],{"class":1231},[1050,9304,1089],{"class":1088},[1050,9306,5866],{"class":1126},[1050,9308,4614],{"class":1088},[1050,9310,9311,9313,9315,9317,9319,9321,9323,9325],{"class":1052,"line":1992},[1050,9312,9216],{"class":4634},[1050,9314,1118],{"class":1088},[1050,9316,1192],{"class":1225},[1050,9318,1089],{"class":1088},[1050,9320,2938],{"class":1126},[1050,9322,1130],{"class":1088},[1050,9324,1366],{"class":1126},[1050,9326,2866],{"class":1088},[1050,9328,9329,9332,9334,9336,9338,9340],{"class":1052,"line":2002},[1050,9330,9331],{"class":4634},"                path",[1050,9333,1118],{"class":1088},[1050,9335,1366],{"class":1126},[1050,9337,1089],{"class":1088},[1050,9339,1752],{"class":1231},[1050,9341,2485],{"class":1088},[1050,9343,9344,9347,9349,9351,9353,9355],{"class":1052,"line":2027},[1050,9345,9346],{"class":4634},"                method",[1050,9348,1118],{"class":1088},[1050,9350,1366],{"class":1126},[1050,9352,1089],{"class":1088},[1050,9354,2809],{"class":1231},[1050,9356,2485],{"class":1088},[1050,9358,9359,9362,9364,9366,9368,9370,9372,9374,9376,9378,9380,9382,9384,9386],{"class":1052,"line":2072},[1050,9360,9361],{"class":4634},"                user_agent",[1050,9363,1118],{"class":1088},[1050,9365,1366],{"class":1126},[1050,9367,1089],{"class":1088},[1050,9369,2962],{"class":2862},[1050,9371,1089],{"class":1088},[1050,9373,2967],{"class":1126},[1050,9375,1130],{"class":1088},[1050,9377,1436],{"class":1260},[1050,9379,2974],{"class":1264},[1050,9381,1436],{"class":1260},[1050,9383,1195],{"class":1088},[1050,9385,2981],{"class":1260},[1050,9387,2866],{"class":1088},[1050,9389,9390,9393,9395,9397,9399,9401,9403,9405,9407,9409,9411,9413,9415,9417,9419,9421,9423,9425,9427,9429,9432,9434,9436],{"class":1052,"line":2077},[1050,9391,9392],{"class":4634},"                user",[1050,9394,1118],{"class":1088},[1050,9396,1366],{"class":1126},[1050,9398,1089],{"class":1088},[1050,9400,2873],{"class":1231},[1050,9402,2892],{"class":1063},[1050,9404,2008],{"class":1185},[1050,9406,1130],{"class":1088},[1050,9408,1366],{"class":1126},[1050,9410,1195],{"class":1088},[1050,9412,1915],{"class":1260},[1050,9414,2873],{"class":1264},[1050,9416,1436],{"class":1260},[1050,9418,1931],{"class":1088},[1050,9420,3542],{"class":1063},[1050,9422,1291],{"class":1126},[1050,9424,1089],{"class":1088},[1050,9426,2873],{"class":1231},[1050,9428,1089],{"class":1088},[1050,9430,9431],{"class":1231},"is_authenticated",[1050,9433,2911],{"class":1063},[1050,9435,2067],{"class":1849},[1050,9437,2485],{"class":1088},[1050,9439,9440,9443,9445,9448,9450,9452],{"class":1052,"line":2084},[1050,9441,9442],{"class":4634},"                timestamp",[1050,9444,1118],{"class":1088},[1050,9446,9447],{"class":1126},"timezone",[1050,9449,1089],{"class":1088},[1050,9451,2793],{"class":1126},[1050,9453,1334],{"class":1088},[1050,9455,9456],{"class":1052,"line":2089},[1050,9457,9458],{"class":1088},"            )\n",[1050,9460,9461,9463,9465,9467,9469],{"class":1052,"line":2111},[1050,9462,5947],{"class":1063},[1050,9464,5950],{"class":2314},[1050,9466,5953],{"class":1063},[1050,9468,5956],{"class":1067},[1050,9470,1156],{"class":1088},[1050,9472,9473,9475,9477,9479,9481,9483,9486,9488,9490,9492,9494],{"class":1052,"line":2121},[1050,9474,5968],{"class":1067},[1050,9476,1089],{"class":1088},[1050,9478,1894],{"class":1126},[1050,9480,1130],{"class":1088},[1050,9482,1727],{"class":1148},[1050,9484,9485],{"class":1264},"\"Error creating request log: ",[1050,9487,1455],{"class":1454},[1050,9489,5984],{"class":1126},[1050,9491,1464],{"class":1454},[1050,9493,1261],{"class":1264},[1050,9495,1137],{"class":1088},[1050,9497,9498,9500],{"class":1052,"line":2127},[1050,9499,4229],{"class":1063},[1050,9501,1850],{"class":1849},[1050,9503,9504],{"class":1052,"line":2169},[1050,9505,1176],{"class":1067},[1050,9507,9508,9510,9513,9515,9517,9519,9522,9524,9526],{"class":1052,"line":2174},[1050,9509,1182],{"class":1148},[1050,9511,9512],{"class":1644}," update_request_log",[1050,9514,1130],{"class":1088},[1050,9516,1192],{"class":1191},[1050,9518,1195],{"class":1088},[1050,9520,9521],{"class":1198}," request_log",[1050,9523,1195],{"class":1088},[1050,9525,1631],{"class":1198},[1050,9527,1202],{"class":1088},[1050,9529,9530,9532,9535],{"class":1052,"line":2180},[1050,9531,1660],{"class":1162},[1050,9533,9534],{"class":1166},"Update request log with response information",[1050,9536,1170],{"class":1162},[1050,9538,9539,9541,9543],{"class":1052,"line":2201},[1050,9540,2005],{"class":1063},[1050,9542,9521],{"class":1067},[1050,9544,1156],{"class":1088},[1050,9546,9547,9550],{"class":1052,"line":2228},[1050,9548,9549],{"class":1063},"            try",[1050,9551,1156],{"class":1088},[1050,9553,9554,9557,9559,9561,9563,9565,9567],{"class":1052,"line":2259},[1050,9555,9556],{"class":1067},"                request_log",[1050,9558,1089],{"class":1088},[1050,9560,3159],{"class":1231},[1050,9562,1235],{"class":1088},[1050,9564,1631],{"class":1067},[1050,9566,1089],{"class":1088},[1050,9568,5836],{"class":1231},[1050,9570,9571,9573,9575,9578,9580,9582,9584,9586,9588,9590,9592,9594,9596,9598,9600,9602,9604,9606,9608,9610,9612],{"class":1052,"line":2264},[1050,9572,9556],{"class":1067},[1050,9574,1089],{"class":1088},[1050,9576,9577],{"class":1231},"response_size",[1050,9579,1235],{"class":1088},[1050,9581,3215],{"class":1185},[1050,9583,1130],{"class":1088},[1050,9585,2013],{"class":1126},[1050,9587,1089],{"class":1088},[1050,9589,3224],{"class":1231},[1050,9591,1931],{"class":1088},[1050,9593,2892],{"class":1063},[1050,9595,2008],{"class":1185},[1050,9597,1130],{"class":1088},[1050,9599,2013],{"class":1126},[1050,9601,1195],{"class":1088},[1050,9603,1915],{"class":1260},[1050,9605,3224],{"class":1264},[1050,9607,1436],{"class":1260},[1050,9609,1931],{"class":1088},[1050,9611,2911],{"class":1063},[1050,9613,9614],{"class":2331}," 0\n",[1050,9616,9617,9619,9621,9623,9625,9627,9629,9631,9633,9635,9637,9639,9641,9643],{"class":1052,"line":2271},[1050,9618,9556],{"class":1067},[1050,9620,1089],{"class":1088},[1050,9622,5922],{"class":1126},[1050,9624,1130],{"class":1088},[1050,9626,5927],{"class":4634},[1050,9628,5930],{"class":1088},[1050,9630,1436],{"class":1260},[1050,9632,3159],{"class":1264},[1050,9634,1436],{"class":1260},[1050,9636,1195],{"class":1088},[1050,9638,1915],{"class":1260},[1050,9640,9577],{"class":1264},[1050,9642,1436],{"class":1260},[1050,9644,4429],{"class":1088},[1050,9646,9647,9650,9652,9654,9656],{"class":1052,"line":2276},[1050,9648,9649],{"class":1063},"            except",[1050,9651,5950],{"class":2314},[1050,9653,5953],{"class":1063},[1050,9655,5956],{"class":1067},[1050,9657,1156],{"class":1088},[1050,9659,9660,9663,9665,9667,9669,9671,9674,9676,9678,9680,9682],{"class":1052,"line":2290},[1050,9661,9662],{"class":1067},"                logger",[1050,9664,1089],{"class":1088},[1050,9666,1894],{"class":1126},[1050,9668,1130],{"class":1088},[1050,9670,1727],{"class":1148},[1050,9672,9673],{"class":1264},"\"Error updating request log: ",[1050,9675,1455],{"class":1454},[1050,9677,5984],{"class":1126},[1050,9679,1464],{"class":1454},[1050,9681,1261],{"class":1264},[1050,9683,1137],{"class":1088},[1050,9685,9686],{"class":1052,"line":2300},[1050,9687,1176],{"class":1067},[1050,9689,9690,9692,9694,9696,9698,9700,9702],{"class":1052,"line":2309},[1050,9691,1182],{"class":1148},[1050,9693,3306],{"class":1644},[1050,9695,1130],{"class":1088},[1050,9697,1192],{"class":1191},[1050,9699,1195],{"class":1088},[1050,9701,1291],{"class":1198},[1050,9703,1202],{"class":1088},[1050,9705,9706,9708,9710],{"class":1052,"line":3319},[1050,9707,1660],{"class":1162},[1050,9709,5015],{"class":1166},[1050,9711,1170],{"class":1162},[1050,9713,9714,9716,9718,9720,9722,9724,9726,9728,9730,9732,9734,9736],{"class":1052,"line":3329},[1050,9715,3332],{"class":1067},[1050,9717,1118],{"class":1088},[1050,9719,1291],{"class":1067},[1050,9721,1089],{"class":1088},[1050,9723,2962],{"class":2862},[1050,9725,1089],{"class":1088},[1050,9727,2967],{"class":1126},[1050,9729,1130],{"class":1088},[1050,9731,1436],{"class":1260},[1050,9733,3351],{"class":1264},[1050,9735,1436],{"class":1260},[1050,9737,1137],{"class":1088},[1050,9739,9740,9742,9744],{"class":1052,"line":3358},[1050,9741,2005],{"class":1063},[1050,9743,3363],{"class":1067},[1050,9745,1156],{"class":1088},[1050,9747,9748,9750,9752,9754,9756,9758,9760,9762,9764,9766,9768,9770],{"class":1052,"line":3368},[1050,9749,3371],{"class":1067},[1050,9751,1118],{"class":1088},[1050,9753,3363],{"class":1067},[1050,9755,1089],{"class":1088},[1050,9757,3380],{"class":1126},[1050,9759,1130],{"class":1088},[1050,9761,1436],{"class":1260},[1050,9763,1195],{"class":1264},[1050,9765,1436],{"class":1260},[1050,9767,3391],{"class":1088},[1050,9769,3394],{"class":2331},[1050,9771,2335],{"class":1088},[1050,9773,9774,9776],{"class":1052,"line":3399},[1050,9775,3402],{"class":1063},[1050,9777,1156],{"class":1088},[1050,9779,9780,9782,9784,9786,9788,9790,9792,9794,9796,9798,9800,9802],{"class":1052,"line":3407},[1050,9781,3371],{"class":1067},[1050,9783,1118],{"class":1088},[1050,9785,1291],{"class":1067},[1050,9787,1089],{"class":1088},[1050,9789,2962],{"class":2862},[1050,9791,1089],{"class":1088},[1050,9793,2967],{"class":1126},[1050,9795,1130],{"class":1088},[1050,9797,1436],{"class":1260},[1050,9799,3428],{"class":1264},[1050,9801,1436],{"class":1260},[1050,9803,1137],{"class":1088},[1050,9805,9806,9808],{"class":1052,"line":3435},[1050,9807,1478],{"class":1063},[1050,9809,3440],{"class":1067},[1050,9811,9812],{"class":1052,"line":3443},[1050,9813,1109],{"emptyLinePlaceholder":1108},[1050,9815,9816],{"class":1052,"line":3448},[1050,9817,6125],{"class":1056},[1050,9819,9820,9822,9824,9826,9828,9830,9832],{"class":1052,"line":3466},[1050,9821,1149],{"class":1148},[1050,9823,8929],{"class":1152},[1050,9825,1130],{"class":1088},[1050,9827,6138],{"class":6137},[1050,9829,1089],{"class":1088},[1050,9831,6143],{"class":6137},[1050,9833,1202],{"class":1088},[1050,9835,9836,9838,9840,9842,9844,9846],{"class":1052,"line":3476},[1050,9837,6245],{"class":1067},[1050,9839,1118],{"class":1088},[1050,9841,6164],{"class":1067},[1050,9843,1089],{"class":1088},[1050,9845,6254],{"class":1126},[1050,9847,1334],{"class":1088},[1050,9849,9850,9852,9854,9856,9858,9860,9862,9864,9866,9868],{"class":1052,"line":3487},[1050,9851,6195],{"class":1067},[1050,9853,1118],{"class":1088},[1050,9855,6164],{"class":1067},[1050,9857,1089],{"class":1088},[1050,9859,6204],{"class":1126},[1050,9861,1130],{"class":1088},[1050,9863,6209],{"class":4634},[1050,9865,1118],{"class":1088},[1050,9867,6214],{"class":2331},[1050,9869,1137],{"class":1088},[1050,9871,9872,9874,9876,9878,9880,9882,9884,9886,9888,9890],{"class":1052,"line":3518},[1050,9873,6221],{"class":1067},[1050,9875,1118],{"class":1088},[1050,9877,6164],{"class":1067},[1050,9879,1089],{"class":1088},[1050,9881,6204],{"class":1126},[1050,9883,1130],{"class":1088},[1050,9885,6209],{"class":4634},[1050,9887,1118],{"class":1088},[1050,9889,6238],{"class":2331},[1050,9891,1137],{"class":1088},[1050,9893,9894,9896,9898,9900,9902,9904],{"class":1052,"line":3562},[1050,9895,6261],{"class":1067},[1050,9897,1118],{"class":1088},[1050,9899,6164],{"class":1067},[1050,9901,1089],{"class":1088},[1050,9903,6270],{"class":1126},[1050,9905,1334],{"class":1088},[1050,9907,9908,9910,9912,9914,9916,9918,9920,9922,9924,9926,9928,9930,9932,9935,9937,9940,9942,9945,9947,9950,9952,9954],{"class":1052,"line":3608},[1050,9909,6159],{"class":1067},[1050,9911,1118],{"class":1088},[1050,9913,6164],{"class":1067},[1050,9915,1089],{"class":1088},[1050,9917,6169],{"class":1126},[1050,9919,1130],{"class":1088},[1050,9921,6174],{"class":1126},[1050,9923,1195],{"class":1088},[1050,9925,6179],{"class":4634},[1050,9927,1118],{"class":1088},[1050,9929,6138],{"class":1126},[1050,9931,1089],{"class":1088},[1050,9933,9934],{"class":2862},"SET_NULL",[1050,9936,1195],{"class":1088},[1050,9938,9939],{"class":4634}," null",[1050,9941,1118],{"class":1088},[1050,9943,9944],{"class":1849},"True",[1050,9946,1195],{"class":1088},[1050,9948,9949],{"class":4634}," blank",[1050,9951,1118],{"class":1088},[1050,9953,9944],{"class":1849},[1050,9955,1137],{"class":1088},[1050,9957,9958,9960,9962,9964,9966,9968],{"class":1052,"line":3626},[1050,9959,6277],{"class":1067},[1050,9961,1118],{"class":1088},[1050,9963,6164],{"class":1067},[1050,9965,1089],{"class":1088},[1050,9967,6286],{"class":1126},[1050,9969,1334],{"class":1088},[1050,9971,9972,9974,9976,9978,9980,9982,9984,9987,9989,9991,9993,9995,9997,9999],{"class":1052,"line":4695},[1050,9973,6293],{"class":1067},[1050,9975,1118],{"class":1088},[1050,9977,6164],{"class":1067},[1050,9979,1089],{"class":1088},[1050,9981,6302],{"class":1126},[1050,9983,1130],{"class":1088},[1050,9985,9986],{"class":4634},"null",[1050,9988,1118],{"class":1088},[1050,9990,9944],{"class":1849},[1050,9992,1195],{"class":1088},[1050,9994,9949],{"class":4634},[1050,9996,1118],{"class":1088},[1050,9998,9944],{"class":1849},[1050,10000,1137],{"class":1088},[1050,10002,10003,10006,10008,10010,10012,10014,10016,10018,10020,10022,10024,10026,10028,10030],{"class":1052,"line":4700},[1050,10004,10005],{"class":1067},"    response_size ",[1050,10007,1118],{"class":1088},[1050,10009,6164],{"class":1067},[1050,10011,1089],{"class":1088},[1050,10013,6302],{"class":1126},[1050,10015,1130],{"class":1088},[1050,10017,9986],{"class":4634},[1050,10019,1118],{"class":1088},[1050,10021,9944],{"class":1849},[1050,10023,1195],{"class":1088},[1050,10025,9949],{"class":4634},[1050,10027,1118],{"class":1088},[1050,10029,9944],{"class":1849},[1050,10031,1137],{"class":1088},[1050,10033,10034],{"class":1052,"line":4722},[1050,10035,1109],{"emptyLinePlaceholder":1108},[1050,10037,10038,10040,10042,10044,10046,10048,10050],{"class":1052,"line":4732},[1050,10039,1149],{"class":1148},[1050,10041,9200],{"class":1152},[1050,10043,1130],{"class":1088},[1050,10045,6138],{"class":6137},[1050,10047,1089],{"class":1088},[1050,10049,6143],{"class":6137},[1050,10051,1202],{"class":1088},[1050,10053,10054,10056,10058,10060,10062,10064,10066,10069,10071,10073],{"class":1052,"line":4751},[1050,10055,6245],{"class":1067},[1050,10057,1118],{"class":1088},[1050,10059,6164],{"class":1067},[1050,10061,1089],{"class":1088},[1050,10063,6254],{"class":1126},[1050,10065,1130],{"class":1088},[1050,10067,10068],{"class":4634},"unique",[1050,10070,1118],{"class":1088},[1050,10072,9944],{"class":1849},[1050,10074,1137],{"class":1088},[1050,10076,10077,10080,10082,10084,10086,10088,10090,10092,10094,10096],{"class":1052,"line":4762},[1050,10078,10079],{"class":1067},"    reason ",[1050,10081,1118],{"class":1088},[1050,10083,6164],{"class":1067},[1050,10085,1089],{"class":1088},[1050,10087,6204],{"class":1126},[1050,10089,1130],{"class":1088},[1050,10091,6209],{"class":4634},[1050,10093,1118],{"class":1088},[1050,10095,6214],{"class":2331},[1050,10097,1137],{"class":1088},[1050,10099,10100,10103,10105,10107,10109,10111,10113,10116,10118,10120],{"class":1052,"line":4768},[1050,10101,10102],{"class":1067},"    blocked_at ",[1050,10104,1118],{"class":1088},[1050,10106,6164],{"class":1067},[1050,10108,1089],{"class":1088},[1050,10110,6286],{"class":1126},[1050,10112,1130],{"class":1088},[1050,10114,10115],{"class":4634},"auto_now_add",[1050,10117,1118],{"class":1088},[1050,10119,9944],{"class":1849},[1050,10121,1137],{"class":1088},[1050,10123,10124,10127,10129,10131,10133,10136,10138,10141,10143,10145],{"class":1052,"line":4773},[1050,10125,10126],{"class":1067},"    is_active ",[1050,10128,1118],{"class":1088},[1050,10130,6164],{"class":1067},[1050,10132,1089],{"class":1088},[1050,10134,10135],{"class":1126},"BooleanField",[1050,10137,1130],{"class":1088},[1050,10139,10140],{"class":4634},"default",[1050,10142,1118],{"class":1088},[1050,10144,9944],{"class":1849},[1050,10146,1137],{"class":1088},[1030,10148,10150],{"id":10149},"testing-custom-middleware","Testing Custom Middleware",[1035,10152,10154],{"id":10153},"unit-tests-for-middleware","Unit Tests for Middleware",[1040,10156,10158],{"className":1042,"code":10157,"language":1044,"meta":1045,"style":1045},"from django.test import TestCase, RequestFactory\nfrom django.http import HttpResponse\nfrom django.contrib.auth.models import User, AnonymousUser\nfrom myapp.middleware import RequestLoggingMiddleware, RateLimitMiddleware\n\nclass CustomMiddlewareTests(TestCase):\n    \"\"\"Test custom middleware functionality\"\"\"\n    \n    def setUp(self):\n        self.factory = RequestFactory()\n        self.user = User.objects.create_user(\n            username='testuser',\n            email='test@example.com',\n            password='testpass123'\n        )\n    \n    def test_request_logging_middleware(self):\n        \"\"\"Test request logging middleware\"\"\"\n        def get_response(request):\n            return HttpResponse(\"Test response\")\n        \n        middleware = RequestLoggingMiddleware(get_response)\n        request = self.factory.get('/test/')\n        request.user = self.user\n        \n        with self.assertLogs('request_logger', level='INFO') as cm:\n            response = middleware(request)\n        \n        self.assertEqual(response.status_code, 200)\n        self.assertEqual(len(cm.output), 2)  # Request and response logs\n        self.assertIn('REQUEST:', cm.output[0])\n        self.assertIn('RESPONSE:', cm.output[1])\n    \n    def test_rate_limiting_middleware(self):\n        \"\"\"Test rate limiting middleware\"\"\"\n        def get_response(request):\n            return HttpResponse(\"Test response\")\n        \n        middleware = RateLimitMiddleware(get_response)\n        \n        # First request should succeed\n        request = self.factory.get('/api/test/')\n        response = middleware(request)\n        self.assertEqual(response.status_code, 200)\n        \n        # Simulate many requests to trigger rate limit\n        # (This would require mocking the cache for proper testing)\n    \n    def test_mobile_detection_middleware(self):\n        \"\"\"Test mobile detection middleware\"\"\"\n        def get_response(request):\n            return HttpResponse(\"Test response\")\n        \n        middleware = MobileDetectionMiddleware(get_response)\n        \n        # Test mobile user agent\n        request = self.factory.get('/test/')\n        request.META['HTTP_USER_AGENT'] = 'Mozilla/5.0 (iPhone; CPU iPhone OS 14_0 like Mac OS X)'\n        \n        response = middleware(request)\n        \n        self.assertTrue(request.is_mobile)\n        self.assertFalse(request.is_tablet)\n        self.assertFalse(request.is_desktop)\n        self.assertEqual(response['X-Device-Type'], 'mobile')\n",[1047,10159,10160,10181,10196,10223,10244,10248,10262,10271,10275,10288,10304,10327,10343,10359,10373,10377,10381,10394,10403,10416,10434,10438,10453,10481,10498,10502,10545,10561,10565,10589,10623,10655,10686,10690,10703,10712,10724,10740,10744,10758,10762,10767,10794,10808,10830,10834,10839,10844,10848,10861,10870,10882,10898,10902,10916,10920,10925,10951,10978,10982,10996,11000,11019,11038,11056],{"__ignoreMap":1045},[1050,10161,10162,10164,10166,10168,10171,10173,10176,10178],{"class":1052,"line":1053},[1050,10163,1082],{"class":1063},[1050,10165,1085],{"class":1067},[1050,10167,1089],{"class":1088},[1050,10169,10170],{"class":1067},"test ",[1050,10172,1064],{"class":1063},[1050,10174,10175],{"class":1067}," TestCase",[1050,10177,1195],{"class":1088},[1050,10179,10180],{"class":1067}," RequestFactory\n",[1050,10182,10183,10185,10187,10189,10191,10193],{"class":1052,"line":1060},[1050,10184,1082],{"class":1063},[1050,10186,1085],{"class":1067},[1050,10188,1089],{"class":1088},[1050,10190,3672],{"class":1067},[1050,10192,1064],{"class":1063},[1050,10194,10195],{"class":1067}," HttpResponse\n",[1050,10197,10198,10200,10202,10204,10206,10208,10210,10212,10214,10216,10219,10221],{"class":1052,"line":1071},[1050,10199,1082],{"class":1063},[1050,10201,1085],{"class":1067},[1050,10203,1089],{"class":1088},[1050,10205,5141],{"class":1067},[1050,10207,1089],{"class":1088},[1050,10209,5146],{"class":1067},[1050,10211,1089],{"class":1088},[1050,10213,5151],{"class":1067},[1050,10215,1064],{"class":1063},[1050,10217,10218],{"class":1067}," User",[1050,10220,1195],{"class":1088},[1050,10222,5156],{"class":1067},[1050,10224,10225,10227,10230,10232,10235,10237,10239,10241],{"class":1052,"line":1079},[1050,10226,1082],{"class":1063},[1050,10228,10229],{"class":1067}," myapp",[1050,10231,1089],{"class":1088},[1050,10233,10234],{"class":1067},"middleware ",[1050,10236,1064],{"class":1063},[1050,10238,2415],{"class":1067},[1050,10240,1195],{"class":1088},[1050,10242,10243],{"class":1067}," RateLimitMiddleware\n",[1050,10245,10246],{"class":1052,"line":1105},[1050,10247,1109],{"emptyLinePlaceholder":1108},[1050,10249,10250,10252,10255,10257,10260],{"class":1052,"line":1112},[1050,10251,1149],{"class":1148},[1050,10253,10254],{"class":1152}," CustomMiddlewareTests",[1050,10256,1130],{"class":1088},[1050,10258,10259],{"class":6137},"TestCase",[1050,10261,1202],{"class":1088},[1050,10263,10264,10266,10269],{"class":1052,"line":1140},[1050,10265,1163],{"class":1162},[1050,10267,10268],{"class":1166},"Test custom middleware functionality",[1050,10270,1170],{"class":1162},[1050,10272,10273],{"class":1052,"line":1145},[1050,10274,1176],{"class":1067},[1050,10276,10277,10279,10282,10284,10286],{"class":1052,"line":1159},[1050,10278,1182],{"class":1148},[1050,10280,10281],{"class":1644}," setUp",[1050,10283,1130],{"class":1088},[1050,10285,1192],{"class":1191},[1050,10287,1202],{"class":1088},[1050,10289,10290,10292,10294,10297,10299,10302],{"class":1052,"line":1173},[1050,10291,1226],{"class":1225},[1050,10293,1089],{"class":1088},[1050,10295,10296],{"class":1231},"factory",[1050,10298,1235],{"class":1088},[1050,10300,10301],{"class":1126}," RequestFactory",[1050,10303,1334],{"class":1088},[1050,10305,10306,10308,10310,10312,10314,10316,10318,10320,10322,10325],{"class":1052,"line":1179},[1050,10307,1226],{"class":1225},[1050,10309,1089],{"class":1088},[1050,10311,2873],{"class":1231},[1050,10313,1235],{"class":1088},[1050,10315,10218],{"class":1067},[1050,10317,1089],{"class":1088},[1050,10319,5861],{"class":1231},[1050,10321,1089],{"class":1088},[1050,10323,10324],{"class":1126},"create_user",[1050,10326,4614],{"class":1088},[1050,10328,10329,10332,10334,10336,10339,10341],{"class":1052,"line":1205},[1050,10330,10331],{"class":4634},"            username",[1050,10333,1118],{"class":1088},[1050,10335,1436],{"class":1260},[1050,10337,10338],{"class":1264},"testuser",[1050,10340,1436],{"class":1260},[1050,10342,2485],{"class":1088},[1050,10344,10345,10348,10350,10352,10355,10357],{"class":1052,"line":1211},[1050,10346,10347],{"class":4634},"            email",[1050,10349,1118],{"class":1088},[1050,10351,1436],{"class":1260},[1050,10353,10354],{"class":1264},"test@example.com",[1050,10356,1436],{"class":1260},[1050,10358,2485],{"class":1088},[1050,10360,10361,10364,10366,10368,10371],{"class":1052,"line":1217},[1050,10362,10363],{"class":4634},"            password",[1050,10365,1118],{"class":1088},[1050,10367,1436],{"class":1260},[1050,10369,10370],{"class":1264},"testpass123",[1050,10372,2506],{"class":1260},[1050,10374,10375],{"class":1052,"line":1222},[1050,10376,4649],{"class":1088},[1050,10378,10379],{"class":1052,"line":1241},[1050,10380,1176],{"class":1067},[1050,10382,10383,10385,10388,10390,10392],{"class":1052,"line":1247},[1050,10384,1182],{"class":1148},[1050,10386,10387],{"class":1644}," test_request_logging_middleware",[1050,10389,1130],{"class":1088},[1050,10391,1192],{"class":1191},[1050,10393,1202],{"class":1088},[1050,10395,10396,10398,10401],{"class":1052,"line":1272},[1050,10397,1660],{"class":1162},[1050,10399,10400],{"class":1166},"Test request logging middleware",[1050,10402,1170],{"class":1162},[1050,10404,10405,10408,10410,10412,10414],{"class":1052,"line":1277},[1050,10406,10407],{"class":1148},"        def",[1050,10409,1199],{"class":1644},[1050,10411,1130],{"class":1088},[1050,10413,1366],{"class":1198},[1050,10415,1202],{"class":1088},[1050,10417,10418,10420,10423,10425,10427,10430,10432],{"class":1052,"line":1296},[1050,10419,4229],{"class":1063},[1050,10421,10422],{"class":1126}," HttpResponse",[1050,10424,1130],{"class":1088},[1050,10426,1261],{"class":1260},[1050,10428,10429],{"class":1264},"Test response",[1050,10431,1261],{"class":1260},[1050,10433,1137],{"class":1088},[1050,10435,10436],{"class":1052,"line":1301},[1050,10437,1340],{"class":1067},[1050,10439,10440,10443,10445,10447,10449,10451],{"class":1052,"line":1307},[1050,10441,10442],{"class":1067},"        middleware ",[1050,10444,1118],{"class":1088},[1050,10446,2415],{"class":1126},[1050,10448,1130],{"class":1088},[1050,10450,1232],{"class":1126},[1050,10452,1137],{"class":1088},[1050,10454,10455,10458,10460,10462,10464,10466,10468,10470,10472,10474,10477,10479],{"class":1052,"line":1312},[1050,10456,10457],{"class":1067},"        request ",[1050,10459,1118],{"class":1088},[1050,10461,1357],{"class":1225},[1050,10463,1089],{"class":1088},[1050,10465,10296],{"class":1231},[1050,10467,1089],{"class":1088},[1050,10469,2967],{"class":1126},[1050,10471,1130],{"class":1088},[1050,10473,1436],{"class":1260},[1050,10475,10476],{"class":1264},"/test/",[1050,10478,1436],{"class":1260},[1050,10480,1137],{"class":1088},[1050,10482,10483,10485,10487,10489,10491,10493,10495],{"class":1052,"line":1318},[1050,10484,1675],{"class":1067},[1050,10486,1089],{"class":1088},[1050,10488,2873],{"class":1231},[1050,10490,1235],{"class":1088},[1050,10492,1357],{"class":1225},[1050,10494,1089],{"class":1088},[1050,10496,10497],{"class":1231},"user\n",[1050,10499,10500],{"class":1052,"line":1337},[1050,10501,1340],{"class":1067},[1050,10503,10504,10507,10509,10511,10514,10516,10518,10520,10522,10524,10527,10529,10531,10534,10536,10538,10540,10543],{"class":1052,"line":1343},[1050,10505,10506],{"class":1063},"        with",[1050,10508,1357],{"class":1225},[1050,10510,1089],{"class":1088},[1050,10512,10513],{"class":1126},"assertLogs",[1050,10515,1130],{"class":1088},[1050,10517,1436],{"class":1260},[1050,10519,2400],{"class":1264},[1050,10521,1436],{"class":1260},[1050,10523,1195],{"class":1088},[1050,10525,10526],{"class":4634}," level",[1050,10528,1118],{"class":1088},[1050,10530,1436],{"class":1260},[1050,10532,10533],{"class":1264},"INFO",[1050,10535,1436],{"class":1260},[1050,10537,1931],{"class":1088},[1050,10539,5953],{"class":1063},[1050,10541,10542],{"class":1067}," cm",[1050,10544,1156],{"class":1088},[1050,10546,10547,10550,10552,10555,10557,10559],{"class":1052,"line":1349},[1050,10548,10549],{"class":1067},"            response ",[1050,10551,1118],{"class":1088},[1050,10553,10554],{"class":1126}," middleware",[1050,10556,1130],{"class":1088},[1050,10558,1366],{"class":1126},[1050,10560,1137],{"class":1088},[1050,10562,10563],{"class":1052,"line":1371},[1050,10564,1340],{"class":1067},[1050,10566,10567,10569,10571,10574,10576,10578,10580,10582,10584,10587],{"class":1052,"line":1376},[1050,10568,1226],{"class":1225},[1050,10570,1089],{"class":1088},[1050,10572,10573],{"class":1126},"assertEqual",[1050,10575,1130],{"class":1088},[1050,10577,2013],{"class":1126},[1050,10579,1089],{"class":1088},[1050,10581,3159],{"class":1231},[1050,10583,1195],{"class":1088},[1050,10585,10586],{"class":2331}," 200",[1050,10588,1137],{"class":1088},[1050,10590,10591,10593,10595,10597,10599,10602,10604,10607,10609,10612,10615,10618,10620],{"class":1052,"line":1382},[1050,10592,1226],{"class":1225},[1050,10594,1089],{"class":1088},[1050,10596,10573],{"class":1126},[1050,10598,1130],{"class":1088},[1050,10600,10601],{"class":1185},"len",[1050,10603,1130],{"class":1088},[1050,10605,10606],{"class":1126},"cm",[1050,10608,1089],{"class":1088},[1050,10610,10611],{"class":1231},"output",[1050,10613,10614],{"class":1088},"),",[1050,10616,10617],{"class":2331}," 2",[1050,10619,1931],{"class":1088},[1050,10621,10622],{"class":1056},"  # Request and response logs\n",[1050,10624,10625,10627,10629,10632,10634,10636,10639,10641,10643,10645,10647,10649,10651,10653],{"class":1052,"line":1398},[1050,10626,1226],{"class":1225},[1050,10628,1089],{"class":1088},[1050,10630,10631],{"class":1126},"assertIn",[1050,10633,1130],{"class":1088},[1050,10635,1436],{"class":1260},[1050,10637,10638],{"class":1264},"REQUEST:",[1050,10640,1436],{"class":1260},[1050,10642,1195],{"class":1088},[1050,10644,10542],{"class":1126},[1050,10646,1089],{"class":1088},[1050,10648,10611],{"class":1231},[1050,10650,1433],{"class":1088},[1050,10652,3394],{"class":2331},[1050,10654,4429],{"class":1088},[1050,10656,10657,10659,10661,10663,10665,10667,10670,10672,10674,10676,10678,10680,10682,10684],{"class":1052,"line":1416},[1050,10658,1226],{"class":1225},[1050,10660,1089],{"class":1088},[1050,10662,10631],{"class":1126},[1050,10664,1130],{"class":1088},[1050,10666,1436],{"class":1260},[1050,10668,10669],{"class":1264},"RESPONSE:",[1050,10671,1436],{"class":1260},[1050,10673,1195],{"class":1088},[1050,10675,10542],{"class":1126},[1050,10677,1089],{"class":1088},[1050,10679,10611],{"class":1231},[1050,10681,1433],{"class":1088},[1050,10683,4307],{"class":2331},[1050,10685,4429],{"class":1088},[1050,10687,10688],{"class":1052,"line":1421},[1050,10689,1176],{"class":1067},[1050,10691,10692,10694,10697,10699,10701],{"class":1052,"line":1427},[1050,10693,1182],{"class":1148},[1050,10695,10696],{"class":1644}," test_rate_limiting_middleware",[1050,10698,1130],{"class":1088},[1050,10700,1192],{"class":1191},[1050,10702,1202],{"class":1088},[1050,10704,10705,10707,10710],{"class":1052,"line":1470},[1050,10706,1660],{"class":1162},[1050,10708,10709],{"class":1166},"Test rate limiting middleware",[1050,10711,1170],{"class":1162},[1050,10713,10714,10716,10718,10720,10722],{"class":1052,"line":1475},[1050,10715,10407],{"class":1148},[1050,10717,1199],{"class":1644},[1050,10719,1130],{"class":1088},[1050,10721,1366],{"class":1198},[1050,10723,1202],{"class":1088},[1050,10725,10726,10728,10730,10732,10734,10736,10738],{"class":1052,"line":1965},[1050,10727,4229],{"class":1063},[1050,10729,10422],{"class":1126},[1050,10731,1130],{"class":1088},[1050,10733,1261],{"class":1260},[1050,10735,10429],{"class":1264},[1050,10737,1261],{"class":1260},[1050,10739,1137],{"class":1088},[1050,10741,10742],{"class":1052,"line":1970},[1050,10743,1340],{"class":1067},[1050,10745,10746,10748,10750,10752,10754,10756],{"class":1052,"line":1992},[1050,10747,10442],{"class":1067},[1050,10749,1118],{"class":1088},[1050,10751,3694],{"class":1126},[1050,10753,1130],{"class":1088},[1050,10755,1232],{"class":1126},[1050,10757,1137],{"class":1088},[1050,10759,10760],{"class":1052,"line":2002},[1050,10761,1340],{"class":1067},[1050,10763,10764],{"class":1052,"line":2027},[1050,10765,10766],{"class":1056},"        # First request should succeed\n",[1050,10768,10769,10771,10773,10775,10777,10779,10781,10783,10785,10787,10790,10792],{"class":1052,"line":2072},[1050,10770,10457],{"class":1067},[1050,10772,1118],{"class":1088},[1050,10774,1357],{"class":1225},[1050,10776,1089],{"class":1088},[1050,10778,10296],{"class":1231},[1050,10780,1089],{"class":1088},[1050,10782,2967],{"class":1126},[1050,10784,1130],{"class":1088},[1050,10786,1436],{"class":1260},[1050,10788,10789],{"class":1264},"/api/test/",[1050,10791,1436],{"class":1260},[1050,10793,1137],{"class":1088},[1050,10795,10796,10798,10800,10802,10804,10806],{"class":1052,"line":2077},[1050,10797,1352],{"class":1067},[1050,10799,1118],{"class":1088},[1050,10801,10554],{"class":1126},[1050,10803,1130],{"class":1088},[1050,10805,1366],{"class":1126},[1050,10807,1137],{"class":1088},[1050,10809,10810,10812,10814,10816,10818,10820,10822,10824,10826,10828],{"class":1052,"line":2084},[1050,10811,1226],{"class":1225},[1050,10813,1089],{"class":1088},[1050,10815,10573],{"class":1126},[1050,10817,1130],{"class":1088},[1050,10819,2013],{"class":1126},[1050,10821,1089],{"class":1088},[1050,10823,3159],{"class":1231},[1050,10825,1195],{"class":1088},[1050,10827,10586],{"class":2331},[1050,10829,1137],{"class":1088},[1050,10831,10832],{"class":1052,"line":2089},[1050,10833,1340],{"class":1067},[1050,10835,10836],{"class":1052,"line":2111},[1050,10837,10838],{"class":1056},"        # Simulate many requests to trigger rate limit\n",[1050,10840,10841],{"class":1052,"line":2121},[1050,10842,10843],{"class":1056},"        # (This would require mocking the cache for proper testing)\n",[1050,10845,10846],{"class":1052,"line":2127},[1050,10847,1176],{"class":1067},[1050,10849,10850,10852,10855,10857,10859],{"class":1052,"line":2169},[1050,10851,1182],{"class":1148},[1050,10853,10854],{"class":1644}," test_mobile_detection_middleware",[1050,10856,1130],{"class":1088},[1050,10858,1192],{"class":1191},[1050,10860,1202],{"class":1088},[1050,10862,10863,10865,10868],{"class":1052,"line":2174},[1050,10864,1660],{"class":1162},[1050,10866,10867],{"class":1166},"Test mobile detection middleware",[1050,10869,1170],{"class":1162},[1050,10871,10872,10874,10876,10878,10880],{"class":1052,"line":2180},[1050,10873,10407],{"class":1148},[1050,10875,1199],{"class":1644},[1050,10877,1130],{"class":1088},[1050,10879,1366],{"class":1198},[1050,10881,1202],{"class":1088},[1050,10883,10884,10886,10888,10890,10892,10894,10896],{"class":1052,"line":2201},[1050,10885,4229],{"class":1063},[1050,10887,10422],{"class":1126},[1050,10889,1130],{"class":1088},[1050,10891,1261],{"class":1260},[1050,10893,10429],{"class":1264},[1050,10895,1261],{"class":1260},[1050,10897,1137],{"class":1088},[1050,10899,10900],{"class":1052,"line":2228},[1050,10901,1340],{"class":1067},[1050,10903,10904,10906,10908,10910,10912,10914],{"class":1052,"line":2259},[1050,10905,10442],{"class":1067},[1050,10907,1118],{"class":1088},[1050,10909,7183],{"class":1126},[1050,10911,1130],{"class":1088},[1050,10913,1232],{"class":1126},[1050,10915,1137],{"class":1088},[1050,10917,10918],{"class":1052,"line":2264},[1050,10919,1340],{"class":1067},[1050,10921,10922],{"class":1052,"line":2271},[1050,10923,10924],{"class":1056},"        # Test mobile user agent\n",[1050,10926,10927,10929,10931,10933,10935,10937,10939,10941,10943,10945,10947,10949],{"class":1052,"line":2276},[1050,10928,10457],{"class":1067},[1050,10930,1118],{"class":1088},[1050,10932,1357],{"class":1225},[1050,10934,1089],{"class":1088},[1050,10936,10296],{"class":1231},[1050,10938,1089],{"class":1088},[1050,10940,2967],{"class":1126},[1050,10942,1130],{"class":1088},[1050,10944,1436],{"class":1260},[1050,10946,10476],{"class":1264},[1050,10948,1436],{"class":1260},[1050,10950,1137],{"class":1088},[1050,10952,10953,10955,10957,10959,10961,10963,10965,10967,10969,10971,10973,10976],{"class":1052,"line":2290},[1050,10954,1675],{"class":1067},[1050,10956,1089],{"class":1088},[1050,10958,2962],{"class":2862},[1050,10960,1433],{"class":1088},[1050,10962,1436],{"class":1260},[1050,10964,2974],{"class":1264},[1050,10966,1436],{"class":1260},[1050,10968,1444],{"class":1088},[1050,10970,1235],{"class":1088},[1050,10972,1915],{"class":1260},[1050,10974,10975],{"class":1264},"Mozilla/5.0 (iPhone; CPU iPhone OS 14_0 like Mac OS X)",[1050,10977,2506],{"class":1260},[1050,10979,10980],{"class":1052,"line":2300},[1050,10981,1340],{"class":1067},[1050,10983,10984,10986,10988,10990,10992,10994],{"class":1052,"line":2309},[1050,10985,1352],{"class":1067},[1050,10987,1118],{"class":1088},[1050,10989,10554],{"class":1126},[1050,10991,1130],{"class":1088},[1050,10993,1366],{"class":1126},[1050,10995,1137],{"class":1088},[1050,10997,10998],{"class":1052,"line":3319},[1050,10999,1340],{"class":1067},[1050,11001,11002,11004,11006,11009,11011,11013,11015,11017],{"class":1052,"line":3329},[1050,11003,1226],{"class":1225},[1050,11005,1089],{"class":1088},[1050,11007,11008],{"class":1126},"assertTrue",[1050,11010,1130],{"class":1088},[1050,11012,1366],{"class":1126},[1050,11014,1089],{"class":1088},[1050,11016,7622],{"class":1231},[1050,11018,1137],{"class":1088},[1050,11020,11021,11023,11025,11028,11030,11032,11034,11036],{"class":1052,"line":3358},[1050,11022,1226],{"class":1225},[1050,11024,1089],{"class":1088},[1050,11026,11027],{"class":1126},"assertFalse",[1050,11029,1130],{"class":1088},[1050,11031,1366],{"class":1126},[1050,11033,1089],{"class":1088},[1050,11035,7645],{"class":1231},[1050,11037,1137],{"class":1088},[1050,11039,11040,11042,11044,11046,11048,11050,11052,11054],{"class":1052,"line":3368},[1050,11041,1226],{"class":1225},[1050,11043,1089],{"class":1088},[1050,11045,11027],{"class":1126},[1050,11047,1130],{"class":1088},[1050,11049,1366],{"class":1126},[1050,11051,1089],{"class":1088},[1050,11053,7668],{"class":1231},[1050,11055,1137],{"class":1088},[1050,11057,11058,11060,11062,11064,11066,11068,11070,11072,11074,11076,11079,11081,11083,11085],{"class":1052,"line":3399},[1050,11059,1226],{"class":1225},[1050,11061,1089],{"class":1088},[1050,11063,10573],{"class":1126},[1050,11065,1130],{"class":1088},[1050,11067,2013],{"class":1126},[1050,11069,1433],{"class":1088},[1050,11071,1436],{"class":1260},[1050,11073,7821],{"class":1264},[1050,11075,1436],{"class":1260},[1050,11077,11078],{"class":1088},"],",[1050,11080,1915],{"class":1260},[1050,11082,8058],{"class":1264},[1050,11084,1436],{"class":1260},[1050,11086,1137],{"class":1088},[1035,11088,11090],{"id":11089},"integration-tests","Integration Tests",[1040,11092,11094],{"className":1042,"code":11093,"language":1044,"meta":1045,"style":1045},"from django.test import TestCase, Client\nfrom django.urls import reverse\n\nclass MiddlewareIntegrationTests(TestCase):\n    \"\"\"Integration tests for middleware\"\"\"\n    \n    def setUp(self):\n        self.client = Client()\n    \n    def test_middleware_chain_execution(self):\n        \"\"\"Test that middleware chain executes correctly\"\"\"\n        response = self.client.get('/test/')\n        \n        # Check that all middleware added their headers\n        self.assertIn('X-Processing-Time', response)\n        self.assertIn('X-Request-ID', response)\n        self.assertIn('X-Device-Type', response)\n    \n    def test_middleware_with_authentication(self):\n        \"\"\"Test middleware behavior with authenticated users\"\"\"\n        user = User.objects.create_user(\n            username='testuser',\n            password='testpass123'\n        )\n        \n        self.client.login(username='testuser', password='testpass123')\n        response = self.client.get('/dashboard/')\n        \n        # Check that user activity was tracked\n        self.assertTrue(UserActivity.objects.filter(user=user).exists())\n",[1047,11095,11096,11115,11131,11135,11148,11157,11161,11173,11189,11193,11206,11215,11241,11245,11250,11272,11294,11316,11320,11333,11342,11361,11375,11387,11391,11395,11436,11462,11466,11471],{"__ignoreMap":1045},[1050,11097,11098,11100,11102,11104,11106,11108,11110,11112],{"class":1052,"line":1053},[1050,11099,1082],{"class":1063},[1050,11101,1085],{"class":1067},[1050,11103,1089],{"class":1088},[1050,11105,10170],{"class":1067},[1050,11107,1064],{"class":1063},[1050,11109,10175],{"class":1067},[1050,11111,1195],{"class":1088},[1050,11113,11114],{"class":1067}," Client\n",[1050,11116,11117,11119,11121,11123,11126,11128],{"class":1052,"line":1060},[1050,11118,1082],{"class":1063},[1050,11120,1085],{"class":1067},[1050,11122,1089],{"class":1088},[1050,11124,11125],{"class":1067},"urls ",[1050,11127,1064],{"class":1063},[1050,11129,11130],{"class":1067}," reverse\n",[1050,11132,11133],{"class":1052,"line":1071},[1050,11134,1109],{"emptyLinePlaceholder":1108},[1050,11136,11137,11139,11142,11144,11146],{"class":1052,"line":1079},[1050,11138,1149],{"class":1148},[1050,11140,11141],{"class":1152}," MiddlewareIntegrationTests",[1050,11143,1130],{"class":1088},[1050,11145,10259],{"class":6137},[1050,11147,1202],{"class":1088},[1050,11149,11150,11152,11155],{"class":1052,"line":1105},[1050,11151,1163],{"class":1162},[1050,11153,11154],{"class":1166},"Integration tests for middleware",[1050,11156,1170],{"class":1162},[1050,11158,11159],{"class":1052,"line":1112},[1050,11160,1176],{"class":1067},[1050,11162,11163,11165,11167,11169,11171],{"class":1052,"line":1140},[1050,11164,1182],{"class":1148},[1050,11166,10281],{"class":1644},[1050,11168,1130],{"class":1088},[1050,11170,1192],{"class":1191},[1050,11172,1202],{"class":1088},[1050,11174,11175,11177,11179,11182,11184,11187],{"class":1052,"line":1145},[1050,11176,1226],{"class":1225},[1050,11178,1089],{"class":1088},[1050,11180,11181],{"class":1231},"client",[1050,11183,1235],{"class":1088},[1050,11185,11186],{"class":1126}," Client",[1050,11188,1334],{"class":1088},[1050,11190,11191],{"class":1052,"line":1159},[1050,11192,1176],{"class":1067},[1050,11194,11195,11197,11200,11202,11204],{"class":1052,"line":1173},[1050,11196,1182],{"class":1148},[1050,11198,11199],{"class":1644}," test_middleware_chain_execution",[1050,11201,1130],{"class":1088},[1050,11203,1192],{"class":1191},[1050,11205,1202],{"class":1088},[1050,11207,11208,11210,11213],{"class":1052,"line":1179},[1050,11209,1660],{"class":1162},[1050,11211,11212],{"class":1166},"Test that middleware chain executes correctly",[1050,11214,1170],{"class":1162},[1050,11216,11217,11219,11221,11223,11225,11227,11229,11231,11233,11235,11237,11239],{"class":1052,"line":1205},[1050,11218,1352],{"class":1067},[1050,11220,1118],{"class":1088},[1050,11222,1357],{"class":1225},[1050,11224,1089],{"class":1088},[1050,11226,11181],{"class":1231},[1050,11228,1089],{"class":1088},[1050,11230,2967],{"class":1126},[1050,11232,1130],{"class":1088},[1050,11234,1436],{"class":1260},[1050,11236,10476],{"class":1264},[1050,11238,1436],{"class":1260},[1050,11240,1137],{"class":1088},[1050,11242,11243],{"class":1052,"line":1211},[1050,11244,1340],{"class":1067},[1050,11246,11247],{"class":1052,"line":1217},[1050,11248,11249],{"class":1056},"        # Check that all middleware added their headers\n",[1050,11251,11252,11254,11256,11258,11260,11262,11264,11266,11268,11270],{"class":1052,"line":1222},[1050,11253,1226],{"class":1225},[1050,11255,1089],{"class":1088},[1050,11257,10631],{"class":1126},[1050,11259,1130],{"class":1088},[1050,11261,1436],{"class":1260},[1050,11263,1439],{"class":1264},[1050,11265,1436],{"class":1260},[1050,11267,1195],{"class":1088},[1050,11269,1631],{"class":1126},[1050,11271,1137],{"class":1088},[1050,11273,11274,11276,11278,11280,11282,11284,11286,11288,11290,11292],{"class":1052,"line":1241},[1050,11275,1226],{"class":1225},[1050,11277,1089],{"class":1088},[1050,11279,10631],{"class":1126},[1050,11281,1130],{"class":1088},[1050,11283,1436],{"class":1260},[1050,11285,2136],{"class":1264},[1050,11287,1436],{"class":1260},[1050,11289,1195],{"class":1088},[1050,11291,1631],{"class":1126},[1050,11293,1137],{"class":1088},[1050,11295,11296,11298,11300,11302,11304,11306,11308,11310,11312,11314],{"class":1052,"line":1247},[1050,11297,1226],{"class":1225},[1050,11299,1089],{"class":1088},[1050,11301,10631],{"class":1126},[1050,11303,1130],{"class":1088},[1050,11305,1436],{"class":1260},[1050,11307,7821],{"class":1264},[1050,11309,1436],{"class":1260},[1050,11311,1195],{"class":1088},[1050,11313,1631],{"class":1126},[1050,11315,1137],{"class":1088},[1050,11317,11318],{"class":1052,"line":1272},[1050,11319,1176],{"class":1067},[1050,11321,11322,11324,11327,11329,11331],{"class":1052,"line":1277},[1050,11323,1182],{"class":1148},[1050,11325,11326],{"class":1644}," test_middleware_with_authentication",[1050,11328,1130],{"class":1088},[1050,11330,1192],{"class":1191},[1050,11332,1202],{"class":1088},[1050,11334,11335,11337,11340],{"class":1052,"line":1296},[1050,11336,1660],{"class":1162},[1050,11338,11339],{"class":1166},"Test middleware behavior with authenticated users",[1050,11341,1170],{"class":1162},[1050,11343,11344,11347,11349,11351,11353,11355,11357,11359],{"class":1052,"line":1301},[1050,11345,11346],{"class":1067},"        user ",[1050,11348,1118],{"class":1088},[1050,11350,10218],{"class":1067},[1050,11352,1089],{"class":1088},[1050,11354,5861],{"class":1231},[1050,11356,1089],{"class":1088},[1050,11358,10324],{"class":1126},[1050,11360,4614],{"class":1088},[1050,11362,11363,11365,11367,11369,11371,11373],{"class":1052,"line":1307},[1050,11364,10331],{"class":4634},[1050,11366,1118],{"class":1088},[1050,11368,1436],{"class":1260},[1050,11370,10338],{"class":1264},[1050,11372,1436],{"class":1260},[1050,11374,2485],{"class":1088},[1050,11376,11377,11379,11381,11383,11385],{"class":1052,"line":1312},[1050,11378,10363],{"class":4634},[1050,11380,1118],{"class":1088},[1050,11382,1436],{"class":1260},[1050,11384,10370],{"class":1264},[1050,11386,2506],{"class":1260},[1050,11388,11389],{"class":1052,"line":1318},[1050,11390,4649],{"class":1088},[1050,11392,11393],{"class":1052,"line":1337},[1050,11394,1340],{"class":1067},[1050,11396,11397,11399,11401,11403,11405,11408,11410,11413,11415,11417,11419,11421,11423,11426,11428,11430,11432,11434],{"class":1052,"line":1343},[1050,11398,1226],{"class":1225},[1050,11400,1089],{"class":1088},[1050,11402,11181],{"class":1231},[1050,11404,1089],{"class":1088},[1050,11406,11407],{"class":1126},"login",[1050,11409,1130],{"class":1088},[1050,11411,11412],{"class":4634},"username",[1050,11414,1118],{"class":1088},[1050,11416,1436],{"class":1260},[1050,11418,10338],{"class":1264},[1050,11420,1436],{"class":1260},[1050,11422,1195],{"class":1088},[1050,11424,11425],{"class":4634}," password",[1050,11427,1118],{"class":1088},[1050,11429,1436],{"class":1260},[1050,11431,10370],{"class":1264},[1050,11433,1436],{"class":1260},[1050,11435,1137],{"class":1088},[1050,11437,11438,11440,11442,11444,11446,11448,11450,11452,11454,11456,11458,11460],{"class":1052,"line":1349},[1050,11439,1352],{"class":1067},[1050,11441,1118],{"class":1088},[1050,11443,1357],{"class":1225},[1050,11445,1089],{"class":1088},[1050,11447,11181],{"class":1231},[1050,11449,1089],{"class":1088},[1050,11451,2967],{"class":1126},[1050,11453,1130],{"class":1088},[1050,11455,1436],{"class":1260},[1050,11457,5257],{"class":1264},[1050,11459,1436],{"class":1260},[1050,11461,1137],{"class":1088},[1050,11463,11464],{"class":1052,"line":1371},[1050,11465,1340],{"class":1067},[1050,11467,11468],{"class":1052,"line":1376},[1050,11469,11470],{"class":1056},"        # Check that user activity was tracked\n",[1050,11472,11473,11475,11477,11479,11481,11484,11486,11488,11490,11492,11494,11496,11498,11500,11502,11504],{"class":1052,"line":1382},[1050,11474,1226],{"class":1225},[1050,11476,1089],{"class":1088},[1050,11478,11008],{"class":1126},[1050,11480,1130],{"class":1088},[1050,11482,11483],{"class":1126},"UserActivity",[1050,11485,1089],{"class":1088},[1050,11487,5861],{"class":1231},[1050,11489,1089],{"class":1088},[1050,11491,9209],{"class":1126},[1050,11493,1130],{"class":1088},[1050,11495,2873],{"class":4634},[1050,11497,1118],{"class":1088},[1050,11499,2873],{"class":1126},[1050,11501,3600],{"class":1088},[1050,11503,9240],{"class":1126},[1050,11505,4073],{"class":1088},[1030,11507,11509],{"id":11508},"middleware-configuration","Middleware Configuration",[1035,11511,11513],{"id":11512},"settings-configuration","Settings Configuration",[1040,11515,11517],{"className":1042,"code":11516,"language":1044,"meta":1045,"style":1045},"# settings.py\n\n# Add custom middleware to MIDDLEWARE list\nMIDDLEWARE = [\n    'django.middleware.security.SecurityMiddleware',\n    'myapp.middleware.RequestLoggingMiddleware',\n    'myapp.middleware.RateLimitMiddleware',\n    'django.contrib.sessions.middleware.SessionMiddleware',\n    'django.middleware.common.CommonMiddleware',\n    'django.middleware.csrf.CsrfViewMiddleware',\n    'django.contrib.auth.middleware.AuthenticationMiddleware',\n    'myapp.middleware.UserActivityMiddleware',\n    'django.contrib.messages.middleware.MessageMiddleware',\n    'myapp.middleware.MobileDetectionMiddleware',\n    'django.middleware.clickjacking.XFrameOptionsMiddleware',\n]\n\n# Custom middleware configuration\nCONDITIONAL_MIDDLEWARE_CONFIG = {\n    'enabled': True,\n    'debug': False,\n    'allowed_ips': ['127.0.0.1', '192.168.1.0/24'],\n    'excluded_paths': ['/static/', '/media/', '/health/']\n}\n\n# Logging configuration for middleware\nLOGGING = {\n    'version': 1,\n    'disable_existing_loggers': False,\n    'handlers': {\n        'request_file': {\n            'level': 'INFO',\n            'class': 'logging.handlers.RotatingFileHandler',\n            'filename': 'logs/requests.log',\n            'maxBytes': 10 * 1024 * 1024,  # 10MB\n            'backupCount': 5,\n        },\n    },\n    'loggers': {\n        'request_logger': {\n            'handlers': ['request_file'],\n            'level': 'INFO',\n            'propagate': False,\n        },\n    },\n}\n",[1047,11518,11519,11524,11528,11533,11542,11554,11565,11576,11587,11598,11609,11620,11631,11642,11653,11664,11668,11672,11677,11685,11699,11713,11743,11780,11785,11789,11794,11803,11818,11833,11846,11860,11879,11898,11918,11946,11962,11967,11972,11985,11997,12017,12035,12050,12054,12058],{"__ignoreMap":1045},[1050,11520,11521],{"class":1052,"line":1053},[1050,11522,11523],{"class":1056},"# settings.py\n",[1050,11525,11526],{"class":1052,"line":1060},[1050,11527,1109],{"emptyLinePlaceholder":1108},[1050,11529,11530],{"class":1052,"line":1071},[1050,11531,11532],{"class":1056},"# Add custom middleware to MIDDLEWARE list\n",[1050,11534,11535,11538,11540],{"class":1052,"line":1079},[1050,11536,11537],{"class":1225},"MIDDLEWARE",[1050,11539,1235],{"class":1088},[1050,11541,2472],{"class":1088},[1050,11543,11544,11547,11550,11552],{"class":1052,"line":1105},[1050,11545,11546],{"class":1260},"    '",[1050,11548,11549],{"class":1264},"django.middleware.security.SecurityMiddleware",[1050,11551,1436],{"class":1260},[1050,11553,2485],{"class":1088},[1050,11555,11556,11558,11561,11563],{"class":1052,"line":1112},[1050,11557,11546],{"class":1260},[1050,11559,11560],{"class":1264},"myapp.middleware.RequestLoggingMiddleware",[1050,11562,1436],{"class":1260},[1050,11564,2485],{"class":1088},[1050,11566,11567,11569,11572,11574],{"class":1052,"line":1140},[1050,11568,11546],{"class":1260},[1050,11570,11571],{"class":1264},"myapp.middleware.RateLimitMiddleware",[1050,11573,1436],{"class":1260},[1050,11575,2485],{"class":1088},[1050,11577,11578,11580,11583,11585],{"class":1052,"line":1145},[1050,11579,11546],{"class":1260},[1050,11581,11582],{"class":1264},"django.contrib.sessions.middleware.SessionMiddleware",[1050,11584,1436],{"class":1260},[1050,11586,2485],{"class":1088},[1050,11588,11589,11591,11594,11596],{"class":1052,"line":1159},[1050,11590,11546],{"class":1260},[1050,11592,11593],{"class":1264},"django.middleware.common.CommonMiddleware",[1050,11595,1436],{"class":1260},[1050,11597,2485],{"class":1088},[1050,11599,11600,11602,11605,11607],{"class":1052,"line":1173},[1050,11601,11546],{"class":1260},[1050,11603,11604],{"class":1264},"django.middleware.csrf.CsrfViewMiddleware",[1050,11606,1436],{"class":1260},[1050,11608,2485],{"class":1088},[1050,11610,11611,11613,11616,11618],{"class":1052,"line":1179},[1050,11612,11546],{"class":1260},[1050,11614,11615],{"class":1264},"django.contrib.auth.middleware.AuthenticationMiddleware",[1050,11617,1436],{"class":1260},[1050,11619,2485],{"class":1088},[1050,11621,11622,11624,11627,11629],{"class":1052,"line":1205},[1050,11623,11546],{"class":1260},[1050,11625,11626],{"class":1264},"myapp.middleware.UserActivityMiddleware",[1050,11628,1436],{"class":1260},[1050,11630,2485],{"class":1088},[1050,11632,11633,11635,11638,11640],{"class":1052,"line":1211},[1050,11634,11546],{"class":1260},[1050,11636,11637],{"class":1264},"django.contrib.messages.middleware.MessageMiddleware",[1050,11639,1436],{"class":1260},[1050,11641,2485],{"class":1088},[1050,11643,11644,11646,11649,11651],{"class":1052,"line":1217},[1050,11645,11546],{"class":1260},[1050,11647,11648],{"class":1264},"myapp.middleware.MobileDetectionMiddleware",[1050,11650,1436],{"class":1260},[1050,11652,2485],{"class":1088},[1050,11654,11655,11657,11660,11662],{"class":1052,"line":1222},[1050,11656,11546],{"class":1260},[1050,11658,11659],{"class":1264},"django.middleware.clickjacking.XFrameOptionsMiddleware",[1050,11661,1436],{"class":1260},[1050,11663,2485],{"class":1088},[1050,11665,11666],{"class":1052,"line":1241},[1050,11667,2335],{"class":1088},[1050,11669,11670],{"class":1052,"line":1247},[1050,11671,1109],{"emptyLinePlaceholder":1108},[1050,11673,11674],{"class":1052,"line":1272},[1050,11675,11676],{"class":1056},"# Custom middleware configuration\n",[1050,11678,11679,11681,11683],{"class":1052,"line":1277},[1050,11680,8283],{"class":1225},[1050,11682,1235],{"class":1088},[1050,11684,2773],{"class":1088},[1050,11686,11687,11689,11691,11693,11695,11697],{"class":1052,"line":1296},[1050,11688,11546],{"class":1260},[1050,11690,8303],{"class":1264},[1050,11692,1436],{"class":1260},[1050,11694,2785],{"class":1088},[1050,11696,8328],{"class":1849},[1050,11698,2485],{"class":1088},[1050,11700,11701,11703,11705,11707,11709,11711],{"class":1052,"line":1301},[1050,11702,11546],{"class":1260},[1050,11704,8358],{"class":1264},[1050,11706,1436],{"class":1260},[1050,11708,2785],{"class":1088},[1050,11710,8365],{"class":1849},[1050,11712,2485],{"class":1088},[1050,11714,11715,11717,11719,11721,11723,11725,11727,11730,11732,11734,11736,11739,11741],{"class":1052,"line":1307},[1050,11716,11546],{"class":1260},[1050,11718,8376],{"class":1264},[1050,11720,1436],{"class":1260},[1050,11722,2785],{"class":1088},[1050,11724,5252],{"class":1088},[1050,11726,1436],{"class":1260},[1050,11728,11729],{"class":1264},"127.0.0.1",[1050,11731,1436],{"class":1260},[1050,11733,1195],{"class":1088},[1050,11735,1915],{"class":1260},[1050,11737,11738],{"class":1264},"192.168.1.0/24",[1050,11740,1436],{"class":1260},[1050,11742,6500],{"class":1088},[1050,11744,11745,11747,11749,11751,11753,11755,11757,11759,11761,11763,11765,11767,11769,11771,11773,11776,11778],{"class":1052,"line":1312},[1050,11746,11546],{"class":1260},[1050,11748,5287],{"class":1264},[1050,11750,1436],{"class":1260},[1050,11752,2785],{"class":1088},[1050,11754,5252],{"class":1088},[1050,11756,1436],{"class":1260},[1050,11758,5296],{"class":1264},[1050,11760,1436],{"class":1260},[1050,11762,1195],{"class":1088},[1050,11764,1915],{"class":1260},[1050,11766,5305],{"class":1264},[1050,11768,1436],{"class":1260},[1050,11770,1195],{"class":1088},[1050,11772,1915],{"class":1260},[1050,11774,11775],{"class":1264},"/health/",[1050,11777,1436],{"class":1260},[1050,11779,2335],{"class":1088},[1050,11781,11782],{"class":1052,"line":1318},[1050,11783,11784],{"class":1088},"}\n",[1050,11786,11787],{"class":1052,"line":1337},[1050,11788,1109],{"emptyLinePlaceholder":1108},[1050,11790,11791],{"class":1052,"line":1343},[1050,11792,11793],{"class":1056},"# Logging configuration for middleware\n",[1050,11795,11796,11799,11801],{"class":1052,"line":1349},[1050,11797,11798],{"class":1225},"LOGGING",[1050,11800,1235],{"class":1088},[1050,11802,2773],{"class":1088},[1050,11804,11805,11807,11810,11812,11814,11816],{"class":1052,"line":1371},[1050,11806,11546],{"class":1260},[1050,11808,11809],{"class":1264},"version",[1050,11811,1436],{"class":1260},[1050,11813,2785],{"class":1088},[1050,11815,4414],{"class":2331},[1050,11817,2485],{"class":1088},[1050,11819,11820,11822,11825,11827,11829,11831],{"class":1052,"line":1376},[1050,11821,11546],{"class":1260},[1050,11823,11824],{"class":1264},"disable_existing_loggers",[1050,11826,1436],{"class":1260},[1050,11828,2785],{"class":1088},[1050,11830,8365],{"class":1849},[1050,11832,2485],{"class":1088},[1050,11834,11835,11837,11840,11842,11844],{"class":1052,"line":1382},[1050,11836,11546],{"class":1260},[1050,11838,11839],{"class":1264},"handlers",[1050,11841,1436],{"class":1260},[1050,11843,2785],{"class":1088},[1050,11845,2773],{"class":1088},[1050,11847,11848,11851,11854,11856,11858],{"class":1052,"line":1398},[1050,11849,11850],{"class":1260},"        '",[1050,11852,11853],{"class":1264},"request_file",[1050,11855,1436],{"class":1260},[1050,11857,2785],{"class":1088},[1050,11859,2773],{"class":1088},[1050,11861,11862,11864,11867,11869,11871,11873,11875,11877],{"class":1052,"line":1416},[1050,11863,2477],{"class":1260},[1050,11865,11866],{"class":1264},"level",[1050,11868,1436],{"class":1260},[1050,11870,2785],{"class":1088},[1050,11872,1915],{"class":1260},[1050,11874,10533],{"class":1264},[1050,11876,1436],{"class":1260},[1050,11878,2485],{"class":1088},[1050,11880,11881,11883,11885,11887,11889,11891,11894,11896],{"class":1052,"line":1421},[1050,11882,2477],{"class":1260},[1050,11884,1149],{"class":1264},[1050,11886,1436],{"class":1260},[1050,11888,2785],{"class":1088},[1050,11890,1915],{"class":1260},[1050,11892,11893],{"class":1264},"logging.handlers.RotatingFileHandler",[1050,11895,1436],{"class":1260},[1050,11897,2485],{"class":1088},[1050,11899,11900,11902,11905,11907,11909,11911,11914,11916],{"class":1052,"line":1427},[1050,11901,2477],{"class":1260},[1050,11903,11904],{"class":1264},"filename",[1050,11906,1436],{"class":1260},[1050,11908,2785],{"class":1088},[1050,11910,1915],{"class":1260},[1050,11912,11913],{"class":1264},"logs/requests.log",[1050,11915,1436],{"class":1260},[1050,11917,2485],{"class":1088},[1050,11919,11920,11922,11925,11927,11929,11931,11934,11937,11939,11941,11943],{"class":1052,"line":1470},[1050,11921,2477],{"class":1260},[1050,11923,11924],{"class":1264},"maxBytes",[1050,11926,1436],{"class":1260},[1050,11928,2785],{"class":1088},[1050,11930,3820],{"class":2331},[1050,11932,11933],{"class":1409}," *",[1050,11935,11936],{"class":2331}," 1024",[1050,11938,11933],{"class":1409},[1050,11940,11936],{"class":2331},[1050,11942,1195],{"class":1088},[1050,11944,11945],{"class":1056},"  # 10MB\n",[1050,11947,11948,11950,11953,11955,11957,11960],{"class":1052,"line":1475},[1050,11949,2477],{"class":1260},[1050,11951,11952],{"class":1264},"backupCount",[1050,11954,1436],{"class":1260},[1050,11956,2785],{"class":1088},[1050,11958,11959],{"class":2331}," 5",[1050,11961,2485],{"class":1088},[1050,11963,11964],{"class":1052,"line":1965},[1050,11965,11966],{"class":1088},"        },\n",[1050,11968,11969],{"class":1052,"line":1970},[1050,11970,11971],{"class":1088},"    },\n",[1050,11973,11974,11976,11979,11981,11983],{"class":1052,"line":1992},[1050,11975,11546],{"class":1260},[1050,11977,11978],{"class":1264},"loggers",[1050,11980,1436],{"class":1260},[1050,11982,2785],{"class":1088},[1050,11984,2773],{"class":1088},[1050,11986,11987,11989,11991,11993,11995],{"class":1052,"line":2002},[1050,11988,11850],{"class":1260},[1050,11990,2400],{"class":1264},[1050,11992,1436],{"class":1260},[1050,11994,2785],{"class":1088},[1050,11996,2773],{"class":1088},[1050,11998,11999,12001,12003,12005,12007,12009,12011,12013,12015],{"class":1052,"line":2027},[1050,12000,2477],{"class":1260},[1050,12002,11839],{"class":1264},[1050,12004,1436],{"class":1260},[1050,12006,2785],{"class":1088},[1050,12008,5252],{"class":1088},[1050,12010,1436],{"class":1260},[1050,12012,11853],{"class":1264},[1050,12014,1436],{"class":1260},[1050,12016,6500],{"class":1088},[1050,12018,12019,12021,12023,12025,12027,12029,12031,12033],{"class":1052,"line":2072},[1050,12020,2477],{"class":1260},[1050,12022,11866],{"class":1264},[1050,12024,1436],{"class":1260},[1050,12026,2785],{"class":1088},[1050,12028,1915],{"class":1260},[1050,12030,10533],{"class":1264},[1050,12032,1436],{"class":1260},[1050,12034,2485],{"class":1088},[1050,12036,12037,12039,12042,12044,12046,12048],{"class":1052,"line":2077},[1050,12038,2477],{"class":1260},[1050,12040,12041],{"class":1264},"propagate",[1050,12043,1436],{"class":1260},[1050,12045,2785],{"class":1088},[1050,12047,8365],{"class":1849},[1050,12049,2485],{"class":1088},[1050,12051,12052],{"class":1052,"line":2084},[1050,12053,11966],{"class":1088},[1050,12055,12056],{"class":1052,"line":2089},[1050,12057,11971],{"class":1088},[1050,12059,12060],{"class":1052,"line":2111},[1050,12061,11784],{"class":1088},[1030,12063,12065],{"id":12064},"best-practices-for-custom-middleware","Best Practices for Custom Middleware",[1035,12067,12069],{"id":12068},"performance-considerations","Performance Considerations",[12071,12072,12073,12077,12080,12083],"ul",{},[12074,12075,12076],"li",{},"Keep middleware lightweight and fast",[12074,12078,12079],{},"Avoid database queries in middleware when possible",[12074,12081,12082],{},"Use caching for expensive operations",[12074,12084,12085],{},"Profile middleware performance impact",[1035,12087,12089],{"id":12088},"error-handling","Error Handling",[12071,12091,12092,12095,12098,12101],{},[12074,12093,12094],{},"Always handle exceptions gracefully",[12074,12096,12097],{},"Don't let middleware errors break requests",[12074,12099,12100],{},"Log errors appropriately",[12074,12102,12103],{},"Provide fallback behavior",[1035,12105,541],{"id":12106},"security",[12071,12108,12109,12112,12115,12118],{},[12074,12110,12111],{},"Validate all inputs and outputs",[12074,12113,12114],{},"Be careful with request/response modification",[12074,12116,12117],{},"Implement proper authentication checks",[12074,12119,12120],{},"Log security-relevant events",[1035,12122,12124],{"id":12123},"maintainability","Maintainability",[12071,12126,12127,12130,12133,12136],{},[12074,12128,12129],{},"Keep middleware focused and single-purpose",[12074,12131,12132],{},"Make middleware configurable",[12074,12134,12135],{},"Write comprehensive tests",[12074,12137,12138],{},"Document middleware behavior",[1030,12140,12142],{"id":12141},"next-steps","Next Steps",[1026,12144,12145],{},"Now that you know how to create custom middleware, let's explore middleware ordering and understand how the sequence of middleware affects your application's behavior.",[12147,12148,12149],"style",{},"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 .soVBu, html code.shiki .soVBu{--shiki-light:#39ADB5;--shiki-default:#999999;--shiki-dark:#666666}html pre.shiki code .siWMO, html code.shiki .siWMO{--shiki-light:#6182B8;--shiki-default:#393A34;--shiki-dark:#DBD7CAEE}html pre.shiki code .s131V, html code.shiki .s131V{--shiki-light:#90A4AE;--shiki-default:#998418;--shiki-dark:#B8A965}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 .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 .sJdAF, html code.shiki .sJdAF{--shiki-light:#6182B8;--shiki-default:#998418;--shiki-dark:#B8A965}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 .se3Ec, html code.shiki .se3Ec{--shiki-light:#90A4AE;--shiki-default:#A65E2B;--shiki-dark:#C99076}html pre.shiki code .sBPpx, html code.shiki .sBPpx{--shiki-light:#E53935;--shiki-default:#393A34;--shiki-dark:#DBD7CAEE}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 .sVsLi, html code.shiki .sVsLi{--shiki-light:#39ADB5;--shiki-default:#AB5959;--shiki-dark:#CB7676}html pre.shiki code .s3h35, html code.shiki .s3h35{--shiki-light:#F76D47;--shiki-default:#A65E2B;--shiki-dark:#C99076}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 .sljsM, html code.shiki .sljsM{--shiki-light:#6182B8;--shiki-default:#59873A;--shiki-dark:#80A665}html pre.shiki code .s8XtY, html code.shiki .s8XtY{--shiki-light:#39ADB5;--shiki-default:#1E754F;--shiki-dark:#4D9375}html pre.shiki code .sa2tF, html code.shiki .sa2tF{--shiki-light:#E2931D;--shiki-default:#998418;--shiki-dark:#B8A965}html pre.shiki code .s7CZa, html code.shiki .s7CZa{--shiki-light:#F76D47;--shiki-default:#2F798A;--shiki-dark:#4C9A91}html pre.shiki code .sFGJz, html code.shiki .sFGJz{--shiki-light:#E53935;--shiki-default:#A65E2B;--shiki-dark:#C99076}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 .sYn-s, html code.shiki .sYn-s{--shiki-light:#E2931D;--shiki-default:#59873A;--shiki-dark:#80A665}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}",{"title":1045,"searchDepth":1053,"depth":1060,"links":12151},[12152,12156,12163,12167,12171,12174,12180],{"id":1032,"depth":1060,"text":1033,"children":12153},[12154,12155],{"id":1037,"depth":1071,"text":1038},{"id":1484,"depth":1071,"text":1485},{"id":2338,"depth":1060,"text":2339,"children":12157},[12158,12159,12160,12161,12162],{"id":2342,"depth":1071,"text":2343},{"id":3634,"depth":1071,"text":3635},{"id":5124,"depth":1071,"text":5125},{"id":6408,"depth":1071,"text":6409},{"id":7159,"depth":1071,"text":7160},{"id":8173,"depth":1060,"text":8174,"children":12164},[12165,12166],{"id":8177,"depth":1071,"text":8178},{"id":8910,"depth":1071,"text":8911},{"id":10149,"depth":1060,"text":10150,"children":12168},[12169,12170],{"id":10153,"depth":1071,"text":10154},{"id":11089,"depth":1071,"text":11090},{"id":11508,"depth":1060,"text":11509,"children":12172},[12173],{"id":11512,"depth":1071,"text":11513},{"id":12064,"depth":1060,"text":12065,"children":12175},[12176,12177,12178,12179],{"id":12068,"depth":1071,"text":12069},{"id":12088,"depth":1071,"text":12089},{"id":12106,"depth":1071,"text":541},{"id":12123,"depth":1071,"text":12124},{"id":12141,"depth":1060,"text":12142},"md",null,{},{"title":529,"description":1028},"Y3VsJZFdmev3BTltW9xFqECvYc9Aq6q27Zg6xuHJpHY",[12187,12189],{"title":525,"path":526,"stem":527,"description":12188,"children":-1},"Django comes with several built-in middleware components that handle common web application concerns. Understanding these middleware components helps you leverage Django's capabilities and serves as examples for creating your own middleware.",{"title":533,"path":534,"stem":535,"description":12190,"children":-1},"The order of middleware in Django's MIDDLEWARE setting is crucial for proper application behavior. This chapter explains how middleware ordering affects request/response processing and provides guidelines for optimal middleware arrangement.",1772474936766]