2025-02-10 04:56:46 +00:00
# HTTPZ Web Scanner
2023-12-15 05:02:42 +00:00
2025-02-10 06:14:37 +00:00
data:image/s3,"s3://crabby-images/bf2e3/bf2e350f55f135f007061f8fa899d04afea233c7" alt=""
2025-02-10 04:56:46 +00:00
A high-performance concurrent web scanner written in Python. HTTPZ efficiently scans domains for HTTP/HTTPS services, extracting valuable information like status codes, titles, SSL certificates, and more.
## Requirements
- [Python ](https://www.python.org/downloads/ )
- [aiohttp ](https://pypi.org/project/aiohttp/ )
- [beautifulsoup4 ](https://pypi.org/project/beautifulsoup4/ )
- [cryptography ](https://pypi.org/project/cryptography/ )
- [dnspython ](https://pypi.org/project/dnspython/ )
- [mmh3 ](https://pypi.org/project/mmh3/ )
- [python-dotenv ](https://pypi.org/project/python-dotenv/ )
## Installation
2025-02-11 07:15:39 +00:00
2025-02-12 00:18:52 +00:00
### Via pip *(recommended)*
2025-02-11 07:15:39 +00:00
```bash
# Install from PyPI
2025-02-11 07:55:37 +00:00
pip install httpz_scanner
2025-02-11 07:15:39 +00:00
# The 'httpz' command will now be available in your terminal
httpz --help
```
### From source
2025-02-10 04:56:46 +00:00
```bash
2025-02-11 07:15:39 +00:00
# Clone the repository
2025-02-10 04:56:46 +00:00
git clone https://github.com/acidvegas/httpz
cd httpz
2025-02-11 07:15:39 +00:00
pip install -r requirements.txt
2025-02-10 04:56:46 +00:00
```
2023-12-15 05:02:42 +00:00
2023-12-16 04:02:37 +00:00
## Usage
2025-02-11 07:15:39 +00:00
### Command Line Interface
Basic usage:
```bash
2025-02-11 07:55:37 +00:00
python -m httpz_scanner domains.txt
2025-02-11 07:15:39 +00:00
```
Scan with all flags enabled and output to JSONL:
```bash
2025-02-11 07:55:37 +00:00
python -m httpz_scanner domains.txt -all -c 100 -o results.jsonl -j -p
2025-02-11 07:15:39 +00:00
```
Read from stdin:
```bash
2025-02-11 07:55:37 +00:00
cat domains.txt | python -m httpz_scanner - -all -c 100
echo "example.com" | python -m httpz_scanner - -all
2025-02-11 07:15:39 +00:00
```
Filter by status codes and follow redirects:
```bash
2025-02-11 07:55:37 +00:00
python -m httpz_scanner domains.txt -mc 200,301-399 -ec 404,500 -fr -p
2025-02-11 07:15:39 +00:00
```
Show specific fields with custom timeout and resolvers:
2025-02-10 04:56:46 +00:00
```bash
2025-02-11 07:55:37 +00:00
python -m httpz_scanner domains.txt -sc -ti -i -tls -to 10 -r resolvers.txt
2025-02-11 07:15:39 +00:00
```
Full scan with all options:
```bash
2025-02-11 07:55:37 +00:00
python -m httpz_scanner domains.txt -c 100 -o output.jsonl -j -all -to 10 -mc 200,301 -ec 404,500 -p -ax -r resolvers.txt
2025-02-11 07:15:39 +00:00
```
2025-02-12 00:18:52 +00:00
### Distributed Scanning
Split scanning across multiple machines using the `--shard` argument:
```bash
# Machine 1
httpz domains.txt --shard 1/3
# Machine 2
httpz domains.txt --shard 2/3
# Machine 3
httpz domains.txt --shard 3/3
```
Each machine will process a different subset of domains without overlap. For example, with 3 shards:
- Machine 1 processes lines 0,3,6,9,...
- Machine 2 processes lines 1,4,7,10,...
- Machine 3 processes lines 2,5,8,11,...
This allows efficient distribution of large scans across multiple machines.
2025-02-11 07:15:39 +00:00
### Python Library
```python
import asyncio
2025-02-12 01:57:01 +00:00
import urllib.request
2025-02-11 07:26:48 +00:00
from httpz_scanner import HTTPZScanner
2025-02-11 07:15:39 +00:00
2025-02-12 01:57:01 +00:00
async def scan_from_list() -> list:
with urllib.request.urlopen('https://example.com/domains.txt') as response:
content = response.read().decode()
return [line.strip() for line in content.splitlines() if line.strip()][:20]
async def scan_from_url():
with urllib.request.urlopen('https://example.com/domains.txt') as response:
for line in response:
if line := line.strip():
yield line.decode().strip()
async def scan_from_file():
with open('domains.txt', 'r') as file:
for line in file:
if line := line.strip():
yield line
async def main():
2025-02-11 07:15:39 +00:00
# Initialize scanner with all possible options (showing defaults)
scanner = HTTPZScanner(
2025-02-12 00:18:52 +00:00
concurrent_limit=100, # Number of concurrent requests
2025-02-11 07:15:39 +00:00
timeout=5, # Request timeout in seconds
2025-02-12 00:18:52 +00:00
follow_redirects=False, # Follow redirects (max 10)
2025-02-11 07:15:39 +00:00
check_axfr=False, # Try AXFR transfer against nameservers
resolver_file=None, # Path to custom DNS resolvers file
output_file=None, # Path to JSONL output file
show_progress=False, # Show progress counter
debug_mode=False, # Show error states and debug info
jsonl_output=False, # Output in JSONL format
2025-02-12 00:18:52 +00:00
shard=None, # Tuple of (shard_index, total_shards) for distributed scanning
2025-02-11 07:15:39 +00:00
# Control which fields to show (all False by default unless show_fields is None)
show_fields={
2025-02-12 00:18:52 +00:00
'status_code': True, # Show status code
'content_type': True, # Show content type
'content_length': True, # Show content length
2025-02-11 07:15:39 +00:00
'title': True, # Show page title
'body': True, # Show body preview
'ip': True, # Show IP addresses
'favicon': True, # Show favicon hash
'headers': True, # Show response headers
2025-02-12 00:18:52 +00:00
'follow_redirects': True, # Show redirect chain
2025-02-11 07:15:39 +00:00
'cname': True, # Show CNAME records
'tls': True # Show TLS certificate info
},
# Filter results
2025-02-12 00:18:52 +00:00
match_codes={200,301,302}, # Only show these status codes
exclude_codes={404,500,503} # Exclude these status codes
2025-02-11 07:15:39 +00:00
)
2025-02-12 01:57:01 +00:00
# Example 1: Process file
print('\nProcessing file:')
async for result in scanner.scan(scan_from_file()):
print(f"{result['domain']}: {result['status']}")
2025-02-11 07:15:39 +00:00
2025-02-12 01:57:01 +00:00
# Example 2: Stream URLs
print('\nStreaming URLs:')
async for result in scanner.scan(scan_from_url()):
print(f"{result['domain']}: {result['status']}")
# Example 3: Process list
print('\nProcessing list:')
domains = await scan_from_list()
async for result in scanner.scan(domains):
print(f"{result['domain']}: {result['status']}")
2025-02-11 07:15:39 +00:00
if __name__ == '__main__':
2025-02-12 01:57:01 +00:00
asyncio.run(main())
2025-02-11 07:15:39 +00:00
```
2025-02-12 00:18:52 +00:00
The scanner accepts various input types:
2025-02-12 01:57:01 +00:00
- File paths (string)
2025-02-12 00:18:52 +00:00
- Lists/tuples of domains
- stdin (using '-')
2025-02-12 01:57:01 +00:00
- Async generators that yield domains
2025-02-12 00:18:52 +00:00
2025-02-12 01:57:01 +00:00
All inputs support sharding for distributed scanning using the `shard` parameter.
2025-02-10 04:56:46 +00:00
2025-02-11 07:15:39 +00:00
## Arguments
2025-02-10 04:56:46 +00:00
2025-02-12 00:18:52 +00:00
| Argument | Long Form | Description |
|---------------|------------------|-------------------------------------------------------------|
| `file` | | File containing domains *(one per line)* , use `-` for stdin |
| `-d` | `--debug` | Show error states and debug information |
| `-c N` | `--concurrent N` | Number of concurrent checks *(default: 100)* |
| `-o FILE` | `--output FILE` | Output file path *(JSONL format)* |
| `-j` | `--jsonl` | Output JSON Lines format to console |
| `-all` | `--all-flags` | Enable all output flags |
| `-sh` | `--shard N/T` | Process shard N of T total shards *(e.g., 1/3)* |
2025-02-10 04:56:46 +00:00
### Output Field Flags
| Flag | Long Form | Description |
|--------| ---------------------|----------------------------------|
| `-sc` | `--status-code` | Show status code |
| `-ct` | `--content-type` | Show content type |
| `-ti` | `--title` | Show page title |
| `-b` | `--body` | Show body preview |
| `-i` | `--ip` | Show IP addresses |
| `-f` | `--favicon` | Show favicon hash |
| `-hr` | `--headers` | Show response headers |
| `-cl` | `--content-length` | Show content length |
| `-fr` | `--follow-redirects` | Follow redirects *(max 10)* |
| `-cn` | `--cname` | Show CNAME records |
| `-tls` | `--tls-info` | Show TLS certificate information |
### Other Options
| Option | Long Form | Description |
|-------------|-------------------------|-----------------------------------------------------|
| `-to N` | `--timeout N` | Request timeout in seconds *(default: 5)* |
| `-mc CODES` | `--match-codes CODES` | Only show specific status codes *(comma-separated)* |
| `-ec CODES` | `--exclude-codes CODES` | Exclude specific status codes *(comma-separated)* |
| `-p` | `--progress` | Show progress counter |
2025-02-12 00:18:52 +00:00
| `-ax` | `--axfr` | Try AXFR transfer against nameservers |
| `-r FILE` | `--resolvers FILE` | File containing DNS resolvers *(one per line)* |