百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

如何利用Python开发一个Web网站路径扫描工具?

itomcoil 2024-12-19 13:43 52 浏览

随着互联网技术的普及Web应用程序的增加,如何保证Web网站的安全性成了一个不可被忽视的问题。而Web路径扫描工具是在渗透测试和安全审计中一个比较常见的工具,它可以帮助开发者或者时候安全人员快速的识别到Web应用程序中可能存在漏洞的路径。

路径扫描器的作用就是对目标网站上的文件和目录进行遍历扫描,找到存在敏感信息和敏感资源的路径目录,下面我们就来详细介绍一下如何通过Python开发一个简单的网站路径扫描工具。

需求分析

在开发之前,首先明确需求,首先需要根据用户提供的网站的URL,我们需要构建出不同的网站路径组合,然后通过发送HTTP请求来检查这些路径是否存在并且是否存在可疑信息,另外就是由于扫描的路径会很多,所以需要用到多线程相关的知识点来提高扫描的速度避免单线程扫描导致的时间过长问题。

环境搭建

根据需求分析,我们可以知道,需要用到如下的一些Python第三方开发库。

  • requests:用于发送HTTP请求,检查路径是否存在。
  • threading:实现多线程,提高扫描效率。
  • argparse:解析命令行参数,方便用户配置扫描任务。
  • os:用于处理文件路径和创建文件。

我们可以通过pip命令来安装对应的第三方开发库。

pip install requests

基本实现流程

构建URL

根据给定的域名和路径字典构建URL,进行路径扫描,如下所示。

url = f"{self.base_url}/{path}"

发送HTTP请求

通过requests提供的get()方法向构建的URL发送请求,检查路径是否存在,如下所示。

response = requests.get(url)

响应判断

根据HTTP状态码判断路径是否存在。常见的判断标准如下:

  • 200:路径存在。
  • 301/302:路径被重定向。
  • 403:禁止访问,但路径存在。
  • 404:路径不存在。
if response.status_code == 200:
    self._add_to_results(url)
elif response.status_code == 301 or response.status_code == 302:
    self._add_to_results(url, "Redirect")
elif response.status_code == 403:
    self._add_to_results(url, "Forbidden")
elif response.status_code == 404:
    pass  # 路径不存在,不做任何处理

多线程扫描

为了加快扫描的速度,我们可以通过Python的threading库来并发发送请求。

   def scan(self):
        """开始路径扫描"""
        threads = []
        for path in self.wordlist:
            t = threading.Thread(target=self.scan_path, args=(path,))
            threads.append(t)
            if len(threads) >= self.threads:
                for t in threads:
                    t.start()
                for t in threads:
                    t.join()
                threads = []

        # 等待剩余线程完成
        for t in threads:
            t.start()
        for t in threads:
            t.join()

最终将扫描结果保存到文件或终端,如下所示将扫描的结果保存到文件中以便后续分析使用。

    def save_results(self, filename="scan_results.txt"):
        """保存扫描结果到文件"""
        with open(filename, "w") as f:
            for path, status in self.found_paths:
                f.write(f"{path} - {status}\n")
        print(f"Results saved to {filename}")

完整代码

import requests
import threading
import argparse
import os

# 定义扫描类
class PathScanner:
    def __init__(self, base_url, wordlist, threads=10):
        self.base_url = base_url
        self.wordlist = wordlist
        self.threads = threads
        self.lock = threading.Lock()
        self.found_paths = []

    def scan_path(self, path):
        """扫描单个路径,检查路径是否存在"""
        url = f"{self.base_url}/{path}"
        try:
            response = requests.get(url)
            if response.status_code == 200:
                self._add_to_results(url)
            elif response.status_code == 301 or response.status_code == 302:
                self._add_to_results(url, "Redirect")
            elif response.status_code == 403:
                self._add_to_results(url, "Forbidden")
            elif response.status_code == 404:
                pass  # 路径不存在,不做任何处理
        except requests.exceptions.RequestException as e:
            print(f"Error scanning {url}: {e}")

    def _add_to_results(self, url, status="Found"):
        """添加扫描结果"""
        with self.lock:
            self.found_paths.append((url, status))

    def scan(self):
        """开始路径扫描"""
        threads = []
        for path in self.wordlist:
            t = threading.Thread(target=self.scan_path, args=(path,))
            threads.append(t)
            if len(threads) >= self.threads:
                for t in threads:
                    t.start()
                for t in threads:
                    t.join()
                threads = []

        # 等待剩余线程完成
        for t in threads:
            t.start()
        for t in threads:
            t.join()

    def save_results(self, filename="scan_results.txt"):
        """保存扫描结果到文件"""
        with open(filename, "w") as f:
            for path, status in self.found_paths:
                f.write(f"{path} - {status}\n")
        print(f"Results saved to {filename}")

# 解析命令行参数
def parse_args():
    parser = argparse.ArgumentParser(description="Web Path Scanner")
    parser.add_argument("url", help="Base URL of the target website")
    parser.add_argument("wordlist", help="Path to the wordlist file")
    parser.add_argument("-t", "--threads", type=int, default=10, help="Number of threads (default: 10)")
    return parser.parse_args()

# 主函数
def main():
    args = parse_args()

    # 读取字典文件
    if not os.path.exists(args.wordlist):
        print(f"Wordlist file {args.wordlist} not found.")
        return

    with open(args.wordlist, "r") as f:
        wordlist = [line.strip() for line in f.readlines()]

    # 创建PathScanner对象并执行扫描
    scanner = PathScanner(args.url, wordlist, args.threads)
    scanner.scan()
    scanner.save_results()

if __name__ == "__main__":
    main()

这里需要注意,由于需要扫描字典,所以我们需要提前利用爬虫技术获取到路径扫描的字典,例如比较常用的一些路径字典如下所示,当然还有很多篇幅有限不做展示。

admin
login
uploads
images
config.php
test.php

最终我们可以通过如下的的命令来运行上面的代码并且实现路径扫描操作。

python path_scanner.py http://example.com wordlist.txt -t 20

其中-t 20表示使用20个线程进行扫描,http://example.com是目标网站的URL,wordlist.txt是路径字典文件。工具会扫描目标站点上是否存在字典文件中的路径,并将结果保存到scan_results.txt文件中。

扩展功能

为了增加扫描的隐蔽性,可以在requests.get()方法中添加代理支持,我们可以通过通过设置proxies参数使用代理。

proxies = {
    "http": "http://10.10.1.10:8080",
    "https": "http://10.10.1.10:8080",
}
response = requests.get(url, proxies=proxies)

可以根据不同的状态码更精确地判断路径的可用性。例如,某些路径可能返回403但实际上是有效的,具体判断可以根据项目需求进一步调整。

总结

本文介绍了如何利用Python开发一个简单的Web路径扫描工具,通过构建URL、发送HTTP请求、并发扫描、结果保存等功能,快速实现了一个基本的扫描工具在实际使用中,用户可以根据自己的需求进一步优化字典文件、代理设置以及扫描策略,使工具更符合特定的渗透测试或安全审计需求。

相关推荐

Excel新函数TEXTSPLIT太强大了,轻松搞定数据拆分!

我是【桃大喵学习记】,欢迎大家关注哟~,每天为你分享职场办公软件使用技巧干货!最近我把WPS软件升级到了版本号:12.1.0.15990的最新版本,最版本已经支持文本拆分函数TEXTSPLIT了,并...

Excel超强数据拆分函数TEXTSPLIT,从入门到精通!

我是【桃大喵学习记】,欢迎大家关注哟~,每天为你分享职场办公软件使用技巧干货!今天跟大家分享的是Excel超强数据拆分函数TEXTSPLIT,带你从入门到精通!TEXTSPLIT函数真是太强大了,轻松...

看完就会用的C++17特性总结(c++11常用新特性)

作者:taoklin,腾讯WXG后台开发一、简单特性1.namespace嵌套C++17使我们可以更加简洁使用命名空间:2.std::variant升级版的C语言Union在C++17之前,通...

plsql字符串分割浅谈(plsql字符集设置)

工作之中遇到的小问题,在此抛出问题,并给出解决方法。一方面是为了给自己留下深刻印象,另一方面给遇到相似问题的同学一个解决思路。如若其中有写的不好或者不对的地方也请不加不吝赐教,集思广益,共同进步。遇到...

javascript如何分割字符串(javascript切割字符串)

javascript如何分割字符串在JavaScript中,您可以使用字符串的`split()`方法来将一个字符串分割成一个数组。`split()`方法接收一个参数,这个参数指定了分割字符串的方式。如...

TextSplit函数的使用方法(入门+进阶+高级共八种用法10个公式)

在Excel和WPS新增的几十个函数中,如果按实用性+功能性排名,textsplit排第二,无函数敢排第一。因为它不仅使用简单,而且解决了以前用超复杂公式才能搞定的难题。今天小编用10个公式,让你彻底...

Python字符串split()方法使用技巧

在Python中,字符串操作可谓是基础且关键的技能,而今天咱们要重点攻克的“堡垒”——split()方法,它能将看似浑然一体的字符串,按照我们的需求进行拆分,极大地便利了数据处理与文本解析工作。基本语...

go语言中字符串常用的系统函数(golang 字符串)

最近由于工作比较忙,视频有段时间没有更新了,在这里跟大家说声抱歉了,我尽快抽些时间整理下视频今天就发一篇关于go语言的基础知识吧!我这我工作中用到的一些常用函数,汇总出来分享给大家,希望对...

无规律文本拆分,这些函数你得会(没有分隔符没规律数据拆分)

今天文章来源于表格学员训练营群内答疑,混合文本拆分。其实拆分不难,只要规则明确就好办。就怕规则不清晰,或者规则太多。那真是,Oh,mygod.如上图所示进行拆分,文字表达实在是有点难,所以小熊变身灵...

Python之文本解析:字符串格式化的逆操作?

引言前面的文章中,提到了关于Python中字符串中的相关操作,更多地涉及到了字符串的格式化,有些地方也称为字符串插值操作,本质上,就是把多个字符串拼接在一起,以固定的格式呈现。关于字符串的操作,其实还...

忘记【分列】吧,TEXTSPLIT拆分文本好用100倍

函数TEXTSPLIT的作用是:按分隔符将字符串拆分为行或列。仅ExcelM365版本可用。基本应用将A2单元格内容按逗号拆分。=TEXTSPLIT(A2,",")第二参数设置为逗号...

Excel365版本新函数TEXTSPLIT,专攻文本拆分

Excel中字符串的处理,拆分和合并是比较常见的需求。合并,当前最好用的函数非TEXTJOIN不可。拆分,Office365于2022年3月更新了一个专业函数:TEXTSPLIT语法参数:【...

站长在线Python精讲使用正则表达式的split()方法分割字符串详解

欢迎你来到站长在线的站长学堂学习Python知识,本文学习的是《在Python中使用正则表达式的split()方法分割字符串详解》。使用正则表达式分割字符串在Python中使用正则表达式的split(...

Java中字符串分割的方法(java字符串切割方法)

技术背景在Java编程中,经常需要对字符串进行分割操作,例如将一个包含多个信息的字符串按照特定的分隔符拆分成多个子字符串。常见的应用场景包括解析CSV文件、处理网络请求参数等。实现步骤1.使用Str...

因为一个函数strtok踩坑,我被老工程师无情嘲笑了

在用C/C++实现字符串切割中,strtok函数经常用到,其主要作用是按照给定的字符集分隔字符串,并返回各子字符串。但是实际上,可不止有strtok(),还有strtok、strtok_s、strto...