当前位置: 首页 > article >正文

Django 学习 Part5、6 :自动化测试与静态文件管理

一、为什么要写测试测试是检查代码操作的例程在不同级别上运行单元测试检查微小细节如模型方法返回值集成测试检查整体操作如用户输入序列是否产生预期结果自动化测试的优势一次创建重复运行修改代码后自动检查是否破坏原有功能无需耗时的人工测试测试哲学你不需要测试整个代码库可以逐步添加测试一个测试总比没有测试好二、发现 Bugwas_published_recently() 的问题我们的Question模型有一个was_published_recently()方法要求如果问题在一天内发布则返回True。但它有个 Bug当pub_date是未来时间时也会返回True在 shell 中验证importdatetimefromdjango.utilsimporttimezonefrompolls.modelsimportQuestion# 创建30天后的未来问题future_questionQuestion(pub_datetimezone.now()datetime.timedelta(days30))future_question.was_published_recently()True# 错误未来时间不应该算最近三、编写第一个测试3.1 创建测试文件在polls/tests.py中编写测试importdatetimefromdjango.testimportTestCasefromdjango.utilsimporttimezonefrom.modelsimportQuestionclassQuestionModelTests(TestCase):deftest_was_published_recently_with_future_question(self): was_published_recently() 对于发布日期在将来的问题应返回 False timetimezone.now()datetime.timedelta(days30)future_questionQuestion(pub_datetime)self.assertIs(future_question.was_published_recently(),False)3.2 运行测试$ python manage.pytestpolls测试执行流程寻找polls应用中的测试代码找到django.test.TestCase的子类创建特殊测试数据库寻找以test开头的方法创建pub_date为30天后的Question实例使用assertIs()检查返回值预期结果测试失败F因为我们尚未修复 Bug四、修复 Bug 并全面测试4.1 修复模型方法修改polls/models.pydefwas_published_recently(self):nowtimezone.now()returnnow-datetime.timedelta(days1)self.pub_datenow4.2 添加全面测试classQuestionModelTests(TestCase):deftest_was_published_recently_with_future_question(self):将来日期返回 Falsetimetimezone.now()datetime.timedelta(days30)future_questionQuestion(pub_datetime)self.assertIs(future_question.was_published_recently(),False)deftest_was_published_recently_with_old_question(self):超过一天的过去日期返回 Falsetimetimezone.now()-datetime.timedelta(days1,seconds1)old_questionQuestion(pub_datetime)self.assertIs(old_question.was_published_recently(),False)deftest_was_published_recently_with_recent_question(self):一天内的日期返回 Truetimetimezone.now()-datetime.timedelta(hours23,minutes59,seconds59)recent_questionQuestion(pub_datetime)self.assertIs(recent_question.was_published_recently(),True)五、测试视图5.1 创建测试工具函数在polls/tests.py中添加defcreate_question(question_text,days): 创建问题的工具函数 days: 负数表示过去正数表示将来 timetimezone.now()datetime.timedelta(daysdays)returnQuestion.objects.create(question_textquestion_text,pub_datetime)5.2 测试 IndexViewfromdjango.urlsimportreverseclassQuestionIndexViewTests(TestCase):deftest_no_questions(self):没有问题时显示适当消息responseself.client.get(reverse(polls:index))self.assertEqual(response.status_code,200)self.assertContains(response,No polls are available.)self.assertQuerySetEqual(response.context[latest_question_list],[])deftest_past_question(self):过去的问题显示在首页past_questioncreate_question(question_textPast question.,days-30)responseself.client.get(reverse(polls:index))self.assertQuerySetEqual(response.context[latest_question_list],[past_question],)deftest_future_question(self):将来的问题不显示在首页create_question(question_textFuture question.,days30)responseself.client.get(reverse(polls:index))self.assertContains(response,No polls are available.)self.assertQuerySetEqual(response.context[latest_question_list],[])deftest_future_question_and_past_question(self):同时存在过去和将来的问题时只显示过去的past_questioncreate_question(question_textPast question.,days-30)create_question(question_textFuture question.,days30)responseself.client.get(reverse(polls:index))self.assertQuerySetEqual(response.context[latest_question_list],[past_question],)deftest_two_past_questions(self):显示多个过去的问题question1create_question(question_textPast question 1.,days-30)question2create_question(question_textPast question 2.,days-5)responseself.client.get(reverse(polls:index))self.assertQuerySetEqual(response.context[latest_question_list],[question2,question1],)5.3 测试 DetailView首先我们需要限制详情页不显示未来的问题。修改polls/views.pyclassDetailView(generic.DetailView):modelQuestion template_namepolls/detail.htmldefget_queryset(self):排除尚未发布的问题returnQuestion.objects.filter(pub_date__ltetimezone.now())然后添加测试classQuestionDetailViewTests(TestCase):deftest_future_question(self):未来问题的详情页返回 404future_questioncreate_question(question_textFuture question.,days5)urlreverse(polls:detail,args(future_question.id,))responseself.client.get(url)self.assertEqual(response.status_code,404)deftest_past_question(self):过去问题的详情页显示问题内容past_questioncreate_question(question_textPast Question.,days-5)urlreverse(polls:detail,args(past_question.id,))responseself.client.get(url)self.assertContains(response,past_question.question_text)第六部分静态文件与样式定制一、静态文件基础1.1 什么是静态文件静态文件是 Web 应用直接返回的内容片段CSS 样式表JavaScript 文件图片文件第三方前端框架1.2 创建静态文件目录在polls目录下创建以下结构polls/ static/ polls/ style.css images/ background.gif命名空间的重要性虽然可以直接放在polls/static/但最佳实践是再创建一层与应用同名的子目录polls/static/polls/。这避免了不同应用中同名静态文件的冲突——Django 只会使用第一个找到的静态文件二、自定义应用样式2.1 创建 CSS 文件创建polls/static/polls/style.css/* 链接样式绿色 */li a{color:green;}/* 页面背景图 */body{background:whiteurl(images/background.gif)no-repeat;}2.2 在模板中引用静态文件修改polls/templates/polls/index.html在文件头部添加{% load static %}linkrelstylesheettypetext/csshref{% static polls/style.css %}关键说明{% load static %}加载静态文件模板标签{% static polls/style.css %}生成静态文件的绝对路径这种方式允许你更改STATIC_URL而无需修改大量静态文件2.3 添加背景图片在polls/static/polls/下创建images目录放入名为background.gif的图片CSS 中已配置body背景图路径使用相对路径images/background.gif警告{% static %}标签在静态文件如 CSS中不可用因为它们不是由 Django 生成的。始终使用相对路径在静态文件之间相互引用三、静态文件配置详解3.1 开发环境配置确保settings.py中包含INSTALLED_APPS[django.contrib.staticfiles,# 确保已启用# ...]STATIC_URL/static/# 访问静态文件的 URL 前缀# 开发时额外查找位置可选STATICFILES_DIRS[BASE_DIR/static,]3.2 查找器机制Django 的STATICFILES_FINDERS设置包含一系列查找器AppDirectoriesFinder在每个INSTALLED_APPS的子目录static中查找四、完整代码汇总polls/tests.py完整版importdatetimefromdjango.testimportTestCasefromdjango.urlsimportreversefromdjango.utilsimporttimezonefrom.modelsimportQuestiondefcreate_question(question_text,days):创建问题的工具函数timetimezone.now()datetime.timedelta(daysdays)returnQuestion.objects.create(question_textquestion_text,pub_datetime)classQuestionModelTests(TestCase):模型测试deftest_was_published_recently_with_future_question(self):将来日期返回 Falsetimetimezone.now()datetime.timedelta(days30)future_questionQuestion(pub_datetime)self.assertIs(future_question.was_published_recently(),False)deftest_was_published_recently_with_old_question(self):超过一天的过去日期返回 Falsetimetimezone.now()-datetime.timedelta(days1,seconds1)old_questionQuestion(pub_datetime)self.assertIs(old_question.was_published_recently(),False)deftest_was_published_recently_with_recent_question(self):一天内的日期返回 Truetimetimezone.now()-datetime.timedelta(hours23,minutes59,seconds59)recent_questionQuestion(pub_datetime)self.assertIs(recent_question.was_published_recently(),True)classQuestionIndexViewTests(TestCase):首页视图测试deftest_no_questions(self):没有问题时显示适当消息responseself.client.get(reverse(polls:index))self.assertEqual(response.status_code,200)self.assertContains(response,No polls are available.)self.assertQuerySetEqual(response.context[latest_question_list],[])deftest_past_question(self):过去的问题显示在首页past_questioncreate_question(question_textPast question.,days-30)responseself.client.get(reverse(polls:index))self.assertQuerySetEqual(response.context[latest_question_list],[past_question],)deftest_future_question(self):将来的问题不显示在首页create_question(question_textFuture question.,days30)responseself.client.get(reverse(polls:index))self.assertContains(response,No polls are available.)self.assertQuerySetEqual(response.context[latest_question_list],[])deftest_future_question_and_past_question(self):同时存在过去和将来的问题时只显示过去的past_questioncreate_question(question_textPast question.,days-30)create_question(question_textFuture question.,days30)responseself.client.get(reverse(polls:index))self.assertQuerySetEqual(response.context[latest_question_list],[past_question],)deftest_two_past_questions(self):显示多个过去的问题按时间排序question1create_question(question_textPast question 1.,days-30)question2create_question(question_textPast question 2.,days-5)responseself.client.get(reverse(polls:index))self.assertQuerySetEqual(response.context[latest_question_list],[question2,question1],)classQuestionDetailViewTests(TestCase):详情页视图测试deftest_future_question(self):未来问题的详情页返回 404future_questioncreate_question(question_textFuture question.,days5)urlreverse(polls:detail,args(future_question.id,))responseself.client.get(url)self.assertEqual(response.status_code,404)deftest_past_question(self):过去问题的详情页显示问题内容past_questioncreate_question(question_textPast Question.,days-5)urlreverse(polls:detail,args(past_question.id,))responseself.client.get(url)self.assertContains(response,past_question.question_text)polls/views.py更新版fromdjango.httpimportHttpResponseRedirectfromdjango.shortcutsimportget_object_or_404,renderfromdjango.urlsimportreversefromdjango.utilsimporttimezonefromdjango.viewsimportgenericfrom.modelsimportChoice,QuestionclassIndexView(generic.ListView):template_namepolls/index.htmlcontext_object_namelatest_question_listdefget_queryset(self):返回最近发布的5个问题不包括未来的returnQuestion.objects.filter(pub_date__ltetimezone.now()).order_by(-pub_date)[:5]classDetailView(generic.DetailView):modelQuestion template_namepolls/detail.htmldefget_queryset(self):排除尚未发布的问题returnQuestion.objects.filter(pub_date__ltetimezone.now())classResultsView(generic.DetailView):modelQuestion template_namepolls/results.htmldefvote(request,question_id):questionget_object_or_404(Question,pkquestion_id)try:selected_choicequestion.choice_set.get(pkrequest.POST[choice])except(KeyError,Choice.DoesNotExist):returnrender(request,polls/detail.html,{question:question,error_message:You didnt select a choice.,})else:selected_choice.votes1selected_choice.save()returnHttpResponseRedirect(reverse(polls:results,args(question.id,)))静态文件结构polls/ static/ polls/ style.css # 主样式表 images/ background.gif # 背景图片 templates/ polls/ index.html # 包含 {% load static %} 和 CSS 引用 detail.html results.html参考资源Django 6.0 官方文档 - 编写你的第一个 Django 应用第 5 部分Django 6.0 官方文档 - 编写你的第一个 Django 应用第 6 部分

相关文章:

Django 学习 Part5、6 :自动化测试与静态文件管理

一、为什么要写测试? 测试是检查代码操作的例程,在不同级别上运行: 单元测试:检查微小细节(如模型方法返回值)集成测试:检查整体操作(如用户输入序列是否产生预期结果)…...

AI 时代,真正拉开人与人差距的,可能是信息来源

AI 时代,真正拉开人与人差距的,可能是信息来源当信息多到看不完时,真正重要的,已经不是你看了多少,而是你到底在看什么。最近,我看到一条很有意思的动态。 Marc Andreessen [硅谷著名投资人,也是…...

小白也能用的Whisper语音识别:上传音频自动转文字实战教程

小白也能用的Whisper语音识别:上传音频自动转文字实战教程 1. 引言:语音转文字原来这么简单 你是否遇到过这些场景: 会议录音需要整理成文字纪要外语视频想快速生成字幕采访录音需要转为可编辑文本 传统方法要么费时费力手动听写&#xf…...

【MCP 2.0安全规范深度解码】:20年协议安全专家逐行剖析RFC草案与OpenMCP参考实现源码

第一章:MCP 2.0安全规范演进脉络与核心设计哲学MCP(Managed Cloud Platform)2.0安全规范并非对1.x版本的简单功能叠加,而是基于零信任架构原则、云原生运行时威胁建模及合规性收敛需求所驱动的范式重构。其演进主线清晰呈现为“从…...

Higress 加入 CNCF:保障 Nginx Ingress 迁移,提供企业级 AI 网关

我们很高兴地宣布,Higress 已正式通过 TOC 投票表决,加入云原生计算基金会(CNCF)Sandbox 项目,成为 CNCF 生态的一员。 Higress 是什么? Higress 是一款基于 Envoy 和 Istio 构建的 AI 原生、高性能 API …...

【MCP协议实战白皮书】:20年架构师亲测——REST API吞吐量下降47%的真相与MCP生产级部署 checklist

第一章:【MCP协议实战白皮书】:20年架构师亲测——REST API吞吐量下降47%的真相与MCP生产级部署 checklist某金融核心系统在接入MCP(Model Control Protocol)协议后,其关键REST API平均吞吐量骤降47%,P95延…...

Gemma-3-270m与STM32开发实战:智能硬件项目

Gemma-3-270m与STM32开发实战:智能硬件项目 1. 项目背景与价值 最近我在做一个智能家居项目,需要让设备能够理解简单的语音指令并做出响应。传统的语音识别方案要么太贵,要么功耗太高,直到我发现了Gemma-3-270m这个超轻量级模型…...

Neeshck-Z-lmage_LYX_v2高效方案:单次生成耗时<12秒的轻量推理优化

Neeshck-Z-lmage_LYX_v2高效方案&#xff1a;单次生成耗时<12秒的轻量推理优化 想体验国产文生图模型&#xff0c;但被复杂的部署、缓慢的生成速度和繁琐的参数调节劝退&#xff1f;今天介绍的这个工具&#xff0c;或许能彻底改变你的看法。Neeshck-Z-lmage_LYX_v2&#xf…...

Fish Speech 1.5语音合成:5分钟快速部署,新手也能玩转多语言TTS

Fish Speech 1.5语音合成&#xff1a;5分钟快速部署&#xff0c;新手也能玩转多语言TTS 1. 从零开始&#xff1a;5分钟搭建你的专属语音合成站 上周有个做知识付费的朋友找我&#xff0c;他想把课程文稿转成多国语言的音频&#xff0c;方便海外学员学习。市面上工具要么贵&am…...

从懵逼到通关:我的第一次 SSH 暴力破解与后门植入实验(小白视角)

大家好&#xff0c;我是刚接触网络安全的小白老许&#xff0c;今天想跟大家聊聊我在靶机环境下完成的第一次渗透实验 —— 全程从上课懵圈到课后一步步啃下来&#xff0c;踩了无数坑&#xff0c;也终于搞懂了「暴力破解」和「持久化后门」到底是怎么回事。⚠️ 重要声明&#x…...

MedGemma应用场景探索:医学教育、科研验证与原型开发

MedGemma应用场景探索&#xff1a;医学教育、科研验证与原型开发 1. 引言&#xff1a;医学AI的多模态革命 在医学影像分析领域&#xff0c;传统方法往往需要专家花费大量时间进行人工解读。MedGemma Medical Vision Lab的出现&#xff0c;为这一过程带来了智能化的解决方案。…...

Z-Image-GGUF惊艳案例集:抽象艺术×中国传统纹样×数字人像融合创作

Z-Image-GGUF惊艳案例集&#xff1a;抽象艺术中国传统纹样数字人像融合创作 1. 项目简介与核心价值 今天我想和你分享一个特别有意思的AI图像生成项目——Z-Image-GGUF。这可不是普通的文生图工具&#xff0c;而是阿里巴巴通义实验室开源模型的一个量化版本&#xff0c;专门为…...

STM32HAL库实战:J-Link RTT高效调试技巧与性能优化

1. J-Link RTT调试技术入门指南 第一次接触J-Link RTT调试技术时&#xff0c;我正被传统串口调试的各种限制困扰着。每次修改代码后都需要重新插拔串口线&#xff0c;调试信息输出速度慢得像老牛拉车&#xff0c;更别提那些因为串口占用导致的资源冲突问题了。直到发现了J-Link…...

Qt 利用QDialog打造动态遮罩层:提升弹窗交互体验

1. 为什么需要动态遮罩层&#xff1f; 做Qt开发的朋友们肯定都遇到过这样的场景&#xff1a;当你点击某个按钮弹出一个对话框时&#xff0c;如果对话框和主窗口的背景色太接近&#xff0c;用户很难快速分辨出对话框的边界。这种情况在深色主题的UI中尤其明显&#xff0c;我曾经…...

WSL2-Debian下CUDA与cuDNN环境配置全攻略

1. WSL2-Debian环境准备 在开始配置CUDA和cuDNN之前&#xff0c;我们需要确保WSL2-Debian环境已经正确设置。WSL2是微软推出的第二代Windows子系统&#xff0c;它提供了接近原生Linux的性能&#xff0c;非常适合开发者和研究人员使用。不过&#xff0c;与原生Linux系统相比&…...

红日靶场(二)Apache与MySQL服务异常排查与修复指南

1. 红日靶场环境简介 红日靶场是网络安全学习者常用的实战演练环境&#xff0c;它模拟了真实的服务器配置和常见漏洞场景。在这个环境中&#xff0c;Apache和MySQL作为基础服务组件&#xff0c;经常会出现各种启动异常问题。我自己在搭建红日靶场时&#xff0c;就遇到过好几次服…...

从计算机组成原理角度看AI模型推理:春联生成的GPU算力消耗

从计算机组成原理角度看AI模型推理&#xff1a;春联生成的GPU算力消耗 春节临近&#xff0c;想用AI模型生成一副独一无二的春联&#xff0c;体验一下科技与传统文化的碰撞。你可能已经试过&#xff0c;输入几个关键词&#xff0c;几秒钟后一副对仗工整、寓意吉祥的春联就跃然屏…...

Windows10环境下DETR模型实战:从零构建自定义数据集训练流程

1. 环境准备与依赖安装 在Windows10上跑通DETR模型的第一步就是搭建合适的开发环境。我建议使用Anaconda创建独立的Python环境&#xff0c;这样可以避免与其他项目的依赖冲突。实测下来Python 3.7版本兼容性最好&#xff0c;PyTorch建议选择1.7.0以上版本。 安装基础依赖时最容…...

实战指南:压控电压源二阶带通滤波器设计与参数优化

1. 压控电压源二阶带通滤波器基础认知 第一次接触压控电压源二阶带通滤波器时&#xff0c;我完全被那些公式和参数搞晕了。后来在实验室熬了三个通宵才明白&#xff0c;这其实就是个"频率筛子"——只让特定频率范围的信号通过&#xff0c;其他频率要么被挡在外面&…...

CTF(misc) USB流量解析实战:从键盘数据到Flag获取

1. USB流量解析在CTF比赛中的重要性 在CTF比赛中&#xff0c;杂项&#xff08;misc&#xff09;类题目往往考验选手的综合分析能力。USB流量解析作为其中的经典题型&#xff0c;近年来频繁出现在各类赛事中。这类题目通常会提供一个包含USB设备通信数据的流量包文件&#xff08…...

Stable Yogi Leather-Dress-Collection生成伦理探讨:建立负责任的AI设计准则

Stable Yogi Leather-Dress-Collection生成伦理探讨&#xff1a;建立负责任的AI设计准则 最近&#xff0c;一个名为Stable Yogi Leather-Dress-Collection的AI设计工具在时尚圈里小火了一把。它能让设计师输入几个关键词&#xff0c;比如“复古机车风”、“鳄鱼纹”、“修身连…...

如何用n8n+Gemini+Pollinations.ai打造小红书爆款笔记全自动生产线

如何用n8nGeminiPollinations.ai打造小红书爆款笔记全自动生产线 在内容为王的时代&#xff0c;小红书运营者每天面临的最大挑战是如何持续产出高质量笔记。传统人工创作模式不仅耗时耗力&#xff0c;更难以保证内容风格的一致性。本文将揭示一套基于n8n工作流引擎的自动化解决…...

ASN.1编码规则实战:从BER到XER的完整解析与避坑指南

ASN.1编码规则实战&#xff1a;从BER到XER的完整解析与避坑指南 在网络协议开发和安全传输领域&#xff0c;数据编码的效率与可靠性直接影响系统性能。ASN.1&#xff08;Abstract Syntax Notation One&#xff09;作为描述数据结构的标准语言&#xff0c;其编码规则的选择往往让…...

赋能中国企业出海:欧洲展台搭建欧标实操解析与孚锐实践

随着中国企业全球化布局持续深化&#xff0c;欧洲作为全球会展业的核心阵地&#xff0c;凭借成熟的行业体系、广阔的市场潜力&#xff0c;成为中国企业出海展示品牌实力的重要舞台。展台作为品牌与欧洲市场对话的核心载体&#xff0c;其搭建质量不仅关乎品牌形象&#xff0c;更…...

DASD-4B-Thinking惊艳效果:gpt-oss-120b知识蒸馏后的4B推理表现

DASD-4B-Thinking惊艳效果&#xff1a;gpt-oss-120b知识蒸馏后的4B推理表现 1. 引言&#xff1a;小模型也能有大智慧 你可能会好奇&#xff0c;一个只有40亿参数的模型&#xff0c;在数学、编程和科学推理这些需要深度思考的任务上&#xff0c;能有什么样的表现&#xff1f;毕…...

STM32G473闪存保护全攻略:PCROP+安全区域配置避坑指南

STM32G473闪存保护全攻略&#xff1a;PCROP安全区域配置避坑指南 在物联网设备开发中&#xff0c;固件和敏感数据的安全保护至关重要。STM32G4系列微控制器提供了多种闪存保护机制&#xff0c;包括专有代码读出保护(PCROP)和安全存储区域配置&#xff0c;这些功能能有效防止未经…...

ComfyUI双PuLID节点工作流排错实录:如何解决KSampler的Float/Half类型冲突问题

ComfyUI双PuLID节点工作流排错实录&#xff1a;如何解决KSampler的Float/Half类型冲突问题 当你在ComfyUI中尝试运行包含两个Apply PuLID Flux节点的复杂工作流时&#xff0c;可能会遇到一个令人困惑的错误&#xff1a;"KSampler expected scalar type Float but found Ha…...

GD32F103上电不启动?5个硬件排查技巧帮你快速定位问题

GD32F103上电不启动&#xff1f;硬件工程师的深度排查实战指南 作为一名常年和GD32F1系列MCU打交道的硬件工程师&#xff0c;我太清楚那种感觉了——电路板焊接完毕&#xff0c;满怀期待地接通电源&#xff0c;结果指示灯不亮&#xff0c;串口没反应&#xff0c;调试器连不上&a…...

AD丝印调整终极指南:从文字居中到批量修改的5个工业级技巧

AD丝印调整终极指南&#xff1a;从文字居中到批量修改的5个工业级技巧 在PCB设计的最后阶段&#xff0c;丝印处理往往成为硬件工程师最容易忽视的环节。那些看似微不足道的白色文字和符号&#xff0c;却是电路板可读性和可维护性的关键所在。想象一下&#xff0c;当你的设计进入…...

CTFshow逆向实战:Base64多层嵌套解码的Python自动化脚本解析(附完整代码)

CTFshow逆向实战&#xff1a;Base64多层嵌套解码的Python自动化脚本解析&#xff08;附完整代码&#xff09; 在CTF竞赛中&#xff0c;Base64编码的嵌套使用是一种常见的混淆手段。本文将深入探讨如何通过Python脚本自动化处理多层Base64嵌套解码问题&#xff0c;帮助参赛者快速…...