forked from kenzuya/unshackle
178 lines
6.9 KiB
Python
178 lines
6.9 KiB
Python
import json
|
|
import random
|
|
import re
|
|
from typing import Optional
|
|
|
|
import requests
|
|
|
|
from unshackle.core.proxies.proxy import Proxy
|
|
|
|
|
|
class WindscribeVPN(Proxy):
|
|
def __init__(self, username: str, password: str, server_map: Optional[dict[str, str]] = None):
|
|
"""
|
|
Proxy Service using WindscribeVPN Service Credentials.
|
|
|
|
A username and password must be provided. These are Service Credentials, not your Login Credentials.
|
|
The Service Credentials can be found here: https://windscribe.com/getconfig/openvpn
|
|
"""
|
|
if not username:
|
|
raise ValueError("No Username was provided to the WindscribeVPN Proxy Service.")
|
|
if not password:
|
|
raise ValueError("No Password was provided to the WindscribeVPN Proxy Service.")
|
|
|
|
if server_map is not None and not isinstance(server_map, dict):
|
|
raise TypeError(f"Expected server_map to be a dict mapping a region to a hostname, not '{server_map!r}'.")
|
|
|
|
self.username = username
|
|
self.password = password
|
|
self.server_map = server_map or {}
|
|
|
|
self.countries = self.get_countries()
|
|
|
|
def __repr__(self) -> str:
|
|
countries = len(set(x.get("country_code") for x in self.countries if x.get("country_code")))
|
|
servers = sum(
|
|
len(host)
|
|
for location in self.countries
|
|
for group in location.get("groups", [])
|
|
for host in group.get("hosts", [])
|
|
)
|
|
|
|
return f"{countries} Countr{['ies', 'y'][countries == 1]} ({servers} Server{['s', ''][servers == 1]})"
|
|
|
|
def get_proxy(self, query: str) -> Optional[str]:
|
|
"""
|
|
Get an HTTPS proxy URI for a WindscribeVPN server.
|
|
|
|
Supports:
|
|
- Country code: "us", "ca", "gb"
|
|
- Specific server: "sg007", "us150"
|
|
- City selection: "us:seattle", "ca:toronto"
|
|
"""
|
|
query = query.lower()
|
|
city = None
|
|
|
|
# Check if query includes city specification (e.g., "ca:toronto")
|
|
if ":" in query:
|
|
query, city = query.split(":", maxsplit=1)
|
|
city = city.strip()
|
|
|
|
# Check server_map for pinned servers (can include city)
|
|
server_map_key = f"{query}:{city}" if city else query
|
|
if server_map_key in self.server_map:
|
|
hostname = self.server_map[server_map_key]
|
|
elif query in self.server_map and not city:
|
|
hostname = self.server_map[query]
|
|
else:
|
|
server_match = re.match(r"^([a-z]{2})(\d+)$", query)
|
|
if server_match:
|
|
# Specific server selection, e.g., sg007, us150
|
|
country_code, server_num = server_match.groups()
|
|
hostname = self.get_specific_server(country_code, server_num)
|
|
if not hostname:
|
|
raise ValueError(
|
|
f"No WindscribeVPN server found matching '{query}'. "
|
|
f"Check the server number or use just '{country_code}' for a random server."
|
|
)
|
|
elif re.match(r"^[a-z]+$", query):
|
|
hostname = self.get_random_server(query, city)
|
|
else:
|
|
raise ValueError(f"The query provided is unsupported and unrecognized: {query}")
|
|
|
|
if not hostname:
|
|
return None
|
|
|
|
hostname = hostname.split(':')[0]
|
|
return f"https://{self.username}:{self.password}@{hostname}:443"
|
|
|
|
def get_specific_server(self, country_code: str, server_num: str) -> Optional[str]:
|
|
"""
|
|
Find a specific server by country code and server number.
|
|
|
|
Matches against hostnames like "sg-007.totallyacdn.com" for query "sg007".
|
|
Tries both the raw number and zero-padded variants.
|
|
|
|
Args:
|
|
country_code: Two-letter country code (e.g., "sg", "us")
|
|
server_num: Server number as string (e.g., "007", "7", "150")
|
|
|
|
Returns:
|
|
The matching hostname, or None if not found.
|
|
"""
|
|
num_stripped = server_num.lstrip("0") or "0"
|
|
candidates = {
|
|
f"{country_code}-{server_num}.",
|
|
f"{country_code}-{num_stripped}.",
|
|
f"{country_code}-{server_num.zfill(3)}.",
|
|
}
|
|
|
|
for location in self.countries:
|
|
if location.get("country_code", "").lower() != country_code:
|
|
continue
|
|
for group in location.get("groups", []):
|
|
for host in group.get("hosts", []):
|
|
hostname = host.get("hostname", "")
|
|
if any(hostname.startswith(prefix) for prefix in candidates):
|
|
return hostname
|
|
|
|
return None
|
|
|
|
def get_random_server(self, country_code: str, city: Optional[str] = None) -> Optional[str]:
|
|
"""
|
|
Get a random server hostname for a country, optionally filtered by city.
|
|
|
|
Args:
|
|
country_code: The country code (e.g., "us", "ca")
|
|
city: Optional city name to filter by (case-insensitive)
|
|
|
|
Returns:
|
|
A random hostname from matching servers, or None if none available.
|
|
"""
|
|
hostnames = []
|
|
|
|
# Collect hostnames from ALL locations matching the country code
|
|
for location in self.countries:
|
|
if location.get("country_code", "").lower() == country_code.lower():
|
|
for group in location.get("groups", []):
|
|
# Filter by city if specified
|
|
if city:
|
|
group_city = group.get("city", "")
|
|
if group_city.lower() != city.lower():
|
|
continue
|
|
|
|
# Collect hostnames from this group
|
|
for host in group.get("hosts", []):
|
|
if hostname := host.get("hostname"):
|
|
hostnames.append(hostname)
|
|
|
|
if hostnames:
|
|
return random.choice(hostnames)
|
|
elif city:
|
|
# No servers found for the specified city
|
|
raise ValueError(
|
|
f"No servers found in city '{city}' for country code '{country_code}'. "
|
|
"Try a different city or check the city name spelling."
|
|
)
|
|
|
|
return None
|
|
|
|
@staticmethod
|
|
def get_countries() -> list[dict]:
|
|
"""Get a list of available Countries and their metadata."""
|
|
res = requests.get(
|
|
url="https://assets.windscribe.com/serverlist/firefox/1/1",
|
|
headers={
|
|
"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/128.0.0.0 Safari/537.36",
|
|
"Content-Type": "application/json",
|
|
},
|
|
)
|
|
if not res.ok:
|
|
raise ValueError(f"Failed to get a list of WindscribeVPN locations [{res.status_code}]")
|
|
|
|
try:
|
|
data = res.json()
|
|
return data.get("data", [])
|
|
except json.JSONDecodeError:
|
|
raise ValueError("Could not decode list of WindscribeVPN locations, not JSON data.")
|