From ee4dd9fd51db3ac76ddb95e3104edc269bebd292 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 15 Feb 2024 06:20:42 +0330 Subject: [PATCH 01/52] feat(cli): added admin usage to table of admins --- cli/admin.py | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/cli/admin.py b/cli/admin.py index 5e2282379..f823a0e9e 100644 --- a/cli/admin.py +++ b/cli/admin.py @@ -5,6 +5,7 @@ from rich.console import Console from rich.panel import Panel from sqlalchemy.exc import IntegrityError +from sqlalchemy.sql import text from decouple import config, UndefinedValueError from app.db import GetDB @@ -34,6 +35,16 @@ def validate_discord_webhook(value: str) -> Union[str, None]: return value +def calculate_admin_usage(admin_id: int) -> int: + with GetDB() as db: + return db.execute(text(f'select sum(used_traffic) / 1024 / 1024 / 1024 from (select * from node_user_usages join (select * from users where admin_id = {admin_id}) as users on users.id = node_user_usages.user_id);')).one()[0] + + +def calculate_admin_reseted_usage(admin_id: int) -> int: + with GetDB() as db: + return db.execute(text(f'select sum(used_traffic_at_reset) / 1024 / 1024 / 1024 from (select * from user_usage_logs join (select * from users where admin_id = {admin_id}) as users on users.id = user_usage_logs.user_id);')).one()[0] + + @app.command(name="list") def list_admins( offset: Optional[int] = typer.Option(None, *utils.FLAGS["offset"]), @@ -44,9 +55,11 @@ def list_admins( with GetDB() as db: admins: list[Admin] = crud.get_admins(db, offset=offset, limit=limit, username=username) utils.print_table( - table=Table("Username", "Is sudo", "Created at", "Telegram ID", "Discord Webhook"), + table=Table("Username", 'Usage', 'Reseted usage', "Is sudo", "Created at", "Telegram ID", "Discord Webhook"), rows=[ (str(admin.username), + f'{calculate_admin_usage(admin.id)}GB', + f'{calculate_admin_reseted_usage(admin.id)}GB', "✔️" if admin.is_sudo else "✖️", utils.readable_datetime(admin.created_at), str(admin.telegram_id or "✖️"), From 31f5fb1a1c82a332232d99b469de2df43b6e562e Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 21 Feb 2024 05:23:01 +0330 Subject: [PATCH 02/52] chore: optimizing query with a little trick --- cli/admin.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/cli/admin.py b/cli/admin.py index f823a0e9e..c773c3c2a 100644 --- a/cli/admin.py +++ b/cli/admin.py @@ -37,12 +37,12 @@ def validate_discord_webhook(value: str) -> Union[str, None]: def calculate_admin_usage(admin_id: int) -> int: with GetDB() as db: - return db.execute(text(f'select sum(used_traffic) / 1024 / 1024 / 1024 from (select * from node_user_usages join (select * from users where admin_id = {admin_id}) as users on users.id = node_user_usages.user_id);')).one()[0] + return db.execute(text(f'SELECT SUM(node_user_usages.used_traffic) / 1024 / 1024 / 1024 FROM node_user_usages JOIN users ON users.id = node_user_usages.user_id AND users.admin_id = {admin_id};')).one()[0] def calculate_admin_reseted_usage(admin_id: int) -> int: with GetDB() as db: - return db.execute(text(f'select sum(used_traffic_at_reset) / 1024 / 1024 / 1024 from (select * from user_usage_logs join (select * from users where admin_id = {admin_id}) as users on users.id = user_usage_logs.user_id);')).one()[0] + return db.execute(text(f'SELECT SUM(used_traffic_at_reset) / 1024 / 1024 / 1024 FROM user_usage_logs JOIN users ON users.id = user_usage_logs.user_id AND users.admin_id = {admin_id};')).one()[0] @app.command(name="list") From 40b1bdabcca91b9d580937366a80a043bc0552a7 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 21 Feb 2024 09:07:58 +0330 Subject: [PATCH 03/52] refactor: use sqlalchemy instead of raw query --- app/db/models.py | 19 ++++++++++++++++--- cli/admin.py | 8 +++++--- 2 files changed, 21 insertions(+), 6 deletions(-) diff --git a/app/db/models.py b/app/db/models.py index 74d59318f..fd905f24c 100644 --- a/app/db/models.py +++ b/app/db/models.py @@ -3,9 +3,10 @@ from sqlalchemy import (JSON, BigInteger, Boolean, Column, DateTime, Enum, Float, ForeignKey, Integer, String, Table, - UniqueConstraint) + UniqueConstraint, func) +from sqlalchemy.ext.hybrid import hybrid_property from sqlalchemy.orm import relationship -from sqlalchemy.sql.expression import text +from sqlalchemy.sql.expression import text, select from app import xray from app.db.base import Base @@ -46,7 +47,7 @@ class User(Base): nullable=False, default=UserDataLimitResetStrategy.no_reset, ) - usage_logs = relationship("UserUsageResetLogs", back_populates="user") + usage_logs = relationship("UserUsageResetLogs", back_populates="user") # maybe rename it to reset_usage_logs? expire = Column(Integer, nullable=True) admin_id = Column(Integer, ForeignKey("admins.id")) admin = relationship("Admin", back_populates="users") @@ -60,6 +61,18 @@ class User(Base): on_hold_timeout = Column(DateTime, nullable=True, default=None) edit_at = Column(DateTime, nullable=True, default=None) + @hybrid_property + def reseted_usage(self): + return sum([log.used_traffic_at_reset for log in self.usage_logs]) + + @reseted_usage.expression + def reseted_usage(self): + return ( + select([func.sum(UserUsageResetLogs.used_traffic_at_reset)]). + where(UserUsageResetLogs.user_id == self.id). + label('reseted_usage') + ) + @property def lifetime_used_traffic(self): return ( diff --git a/cli/admin.py b/cli/admin.py index c773c3c2a..de6c81dd6 100644 --- a/cli/admin.py +++ b/cli/admin.py @@ -5,7 +5,7 @@ from rich.console import Console from rich.panel import Panel from sqlalchemy.exc import IntegrityError -from sqlalchemy.sql import text +from sqlalchemy import func from decouple import config, UndefinedValueError from app.db import GetDB @@ -37,12 +37,14 @@ def validate_discord_webhook(value: str) -> Union[str, None]: def calculate_admin_usage(admin_id: int) -> int: with GetDB() as db: - return db.execute(text(f'SELECT SUM(node_user_usages.used_traffic) / 1024 / 1024 / 1024 FROM node_user_usages JOIN users ON users.id = node_user_usages.user_id AND users.admin_id = {admin_id};')).one()[0] + usage = db.query(func.sum(User.used_traffic)).filter_by(admin_id=admin_id).first()[0] + return 0 if not usage else int(usage / 1024 / 1024 / 1024) # to GB def calculate_admin_reseted_usage(admin_id: int) -> int: with GetDB() as db: - return db.execute(text(f'SELECT SUM(used_traffic_at_reset) / 1024 / 1024 / 1024 FROM user_usage_logs JOIN users ON users.id = user_usage_logs.user_id AND users.admin_id = {admin_id};')).one()[0] + usage = db.query(func.sum(User.reseted_usage)).filter_by(admin_id=admin_id).first()[0] + return 0 if not usage else int(usage / 1024 / 1024 / 1024) # to GB @app.command(name="list") From cff6f6753663ad6e88df30dc10b882a83c5649db Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 15:02:49 +0330 Subject: [PATCH 04/52] fix: report change status temporary --- app/jobs/review_users.py | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/app/jobs/review_users.py b/app/jobs/review_users.py index 6ceb8ce32..7da8d90fa 100644 --- a/app/jobs/review_users.py +++ b/app/jobs/review_users.py @@ -82,10 +82,9 @@ def review(): update_user_status(db, user, status) start_user_expire(db, user) - bg.add_task( - report.status_change, username=user.username, status=status, - user=UserResponse.from_orm(user), user_admin=user.admin - ) + + report.status_change(username=user.username, status=status, + user=UserResponse.from_orm(user), user_admin=user.admin) logger.info(f"User \"{user.username}\" status changed to {status}") From 3d0e3da697c8423092b041158fcde40ae14b887e Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 15:09:39 +0330 Subject: [PATCH 05/52] fix: accept unix domain --- app/xray/config.py | 16 ++++++---------- 1 file changed, 6 insertions(+), 10 deletions(-) diff --git a/app/xray/config.py b/app/xray/config.py index 9759c44e6..8b7e9b6de 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -155,16 +155,12 @@ def _resolve_inbounds(self): try: settings['port'] = inbound['port'] except KeyError: - if not self._fallbacks_inbound: - raise ValueError( - f"port missing on {inbound['tag']}" - "\nset XRAY_FALLBACKS_INBOUND_TAG if you're using an inbound containing fallbacks" - ) - try: - settings['port'] = self._fallbacks_inbound['port'] - settings['is_fallback'] = True - except KeyError: - raise ValueError("fallbacks inbound doesn't have port") + if self._fallbacks_inbound: + try: + settings['port'] = self._fallbacks_inbound['port'] + settings['is_fallback'] = True + except KeyError: + raise ValueError("fallbacks inbound doesn't have port") # stream settings if stream := inbound.get('streamSettings'): From dee8f8170a5efc09c17340314911e8419f0f9c56 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 15:19:12 +0330 Subject: [PATCH 06/52] feat: add spiderx --- app/subscription/v2ray.py | 19 ++++++++++++------- app/xray/config.py | 4 ++++ 2 files changed, 16 insertions(+), 7 deletions(-) diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index 677364843..3e7011f7d 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -39,7 +39,6 @@ def vmess( "net": net, "path": path, "port": port, - "fragment": fs, "ps": remark, "scy": "auto", "tls": tls, @@ -51,6 +50,8 @@ def vmess( payload["sni"] = sni payload["fp"] = fp payload["alpn"] = alpn + if fs : + payload["fragment"] = fs if ais: payload["allowInsecure"] = 1 elif tls == "reality": @@ -111,7 +112,8 @@ def vless(cls, payload["sni"] = sni payload["fp"] = fp payload["alpn"] = alpn - payload["fragment"] = fs + if fs: + payload["fragment"] = fs if ais: payload["allowInsecure"] = 1 elif tls == "reality": @@ -172,7 +174,8 @@ def trojan(cls, payload["sni"] = sni payload["fp"] = fp payload["alpn"] = alpn - payload["fragment"] = fs + if fs: + payload["fragment"] = fs if ais: payload["allowInsecure"] = 1 elif tls == "reality": @@ -236,7 +239,7 @@ def tls_config(sni=None, fp=None, alpn=None, ais=None): return tlsSettings @staticmethod - def reality_config(sni=None, fp=None, pbk=None, sid=None): + def reality_config(sni=None, fp=None, pbk=None, sid=None, spx=None): realitySettings = {} if sni is not None: @@ -250,8 +253,8 @@ def reality_config(sni=None, fp=None, pbk=None, sid=None): realitySettings["publicKey"] = pbk if sid: realitySettings["shortId"] = sid - - realitySettings["spiderX"] = "" + if spx: + realitySettings["spiderX"] = spx return realitySettings @@ -508,6 +511,7 @@ def make_stream_setting(self, alpn='', pbk='', sid='', + spx='', headers='', ais='', dialer_proxy='' @@ -534,7 +538,7 @@ def make_stream_setting(self, tls_settings = self.tls_config(sni=sni, fp=fp, alpn=alpn, ais=ais) elif tls == "reality": tls_settings = self.reality_config( - sni=sni, fp=fp, pbk=pbk, sid=sid) + sni=sni, fp=fp, pbk=pbk, sid=sid, spx=spx) else: tls_settings = None @@ -620,6 +624,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): fp=inbound.get('fp', ''), pbk=inbound.get('pbk', ''), sid=inbound.get('sid', ''), + spx=inbound.get('spx', ''), headers=headers, ais=inbound.get('ais', ''), dialer_proxy=dialer_proxy diff --git a/app/xray/config.py b/app/xray/config.py index 8b7e9b6de..802eed7cd 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -226,6 +226,10 @@ def _resolve_inbounds(self): except (IndexError, TypeError): raise ValueError( f"You need to define at least one shortID in realitySettings of {inbound['tag']}") + try: + settings['spx'] = tls_settings.get('SpiderX') + except: + pass if net == 'tcp': header = net_settings.get('header', {}) From 3d7872bebec4bec789e5bb61bb64581832237ac2 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 15:30:14 +0330 Subject: [PATCH 07/52] fix: days left for expired users --- app/subscription/share.py | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/app/subscription/share.py b/app/subscription/share.py index 48c792ae3..b082c704c 100644 --- a/app/subscription/share.py +++ b/app/subscription/share.py @@ -226,6 +226,8 @@ def setup_format_variables(extra_data: dict) -> dict: user_status = extra_data.get("status") expire_timestamp = extra_data.get("expire") on_hold_expire_duration = extra_data.get("on_hold_expire_duration") + now = dt.utcnow() + now_ts = now.timestamp() if user_status != UserStatus.on_hold: if expire_timestamp is not None and expire_timestamp >= 0: @@ -235,8 +237,13 @@ def setup_format_variables(extra_data: dict) -> dict: jalali_expire_date = jd.fromgregorian( year=expire_date.year, month=expire_date.month, day=expire_date.day ).strftime("%Y-%m-%d") - days_left = (expire_datetime - dt.utcnow()).days + 1 - time_left = format_time_left(seconds_left) + if now_ts < expire_timestamp: + days_left = (expire_datetime - dt.utcnow()).days + 1 + time_left = format_time_left(seconds_left) + else: + days_left = "0" + time_left = "0" + else: days_left = "∞" time_left = "∞" From 12b21b14f87ee2f49f80359f1ca2ce7a77dd5f44 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 15:49:55 +0330 Subject: [PATCH 08/52] feat: add multi mode for grpc --- app/subscription/share.py | 3 +++ app/subscription/v2ray.py | 27 ++++++++++++++++++++++++--- app/xray/config.py | 1 + 3 files changed, 28 insertions(+), 3 deletions(-) diff --git a/app/subscription/share.py b/app/subscription/share.py index b082c704c..e2a4f04be 100644 --- a/app/subscription/share.py +++ b/app/subscription/share.py @@ -53,6 +53,7 @@ def get_v2ray_link(remark: str, address: str, inbound: dict, settings: dict): type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), + mode=inbound.get("multiMode", False), ) if inbound["protocol"] == "vless": @@ -75,6 +76,7 @@ def get_v2ray_link(remark: str, address: str, inbound: dict, settings: dict): type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), + mode=inbound.get("multiMode", False), ) if inbound["protocol"] == "trojan": @@ -97,6 +99,7 @@ def get_v2ray_link(remark: str, address: str, inbound: dict, settings: dict): type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), + mode=inbound.get("multiMode", False), ) if inbound["protocol"] == "shadowsocks": diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index 3e7011f7d..ca6974727 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -30,6 +30,7 @@ def vmess( spx="", ais="", fs="", + multiMode: bool = False, ): payload = { "add": address, @@ -61,6 +62,12 @@ def vmess( payload["sid"] = sid payload["spx"] = spx + if net == "grpc": + if multiMode: + payload["mode"] = "multi" + else: + payload["mode"] = "gun" + return ( "vmess://" + base64.b64encode( @@ -88,6 +95,7 @@ def vless(cls, spx='', ais='', fs="", + multiMode: bool = False, ): payload = { @@ -101,6 +109,11 @@ def vless(cls, if net == 'grpc': payload['serviceName'] = path payload["host"] = host + if multiMode: + payload["mode"] = "multi" + else: + payload["mode"] = "gun" + elif net == 'quic': payload['key'] = path payload["quicSecurity"] = host @@ -150,6 +163,7 @@ def trojan(cls, spx='', ais='', fs="", + multiMode: bool = False, ): payload = { @@ -163,6 +177,11 @@ def trojan(cls, if net == 'grpc': payload['serviceName'] = path payload["host"] = host + if multiMode: + payload["mode"] = "multi" + else: + payload["mode"] = "gun" + elif net == 'quic': payload['key'] = path payload["quicSecurity"] = host @@ -514,13 +533,14 @@ def make_stream_setting(self, spx='', headers='', ais='', - dialer_proxy='' + dialer_proxy='', + multiMode: bool = False, ): if net == "ws": network_setting = self.ws_config(path=path, host=host) elif net == "grpc": - network_setting = self.grpc_config(path=path) + network_setting = self.grpc_config(path=path, multiMode=multiMode) elif net == "h2": network_setting = self.h2_config(path=path, host=host) elif net == "kpc": @@ -627,7 +647,8 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): spx=inbound.get('spx', ''), headers=headers, ais=inbound.get('ais', ''), - dialer_proxy=dialer_proxy + dialer_proxy=dialer_proxy, + multiMode=inbound.get('multiMode',False), ) mux_json = json.loads(self.mux_template) diff --git a/app/xray/config.py b/app/xray/config.py index 802eed7cd..27f648c4a 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -269,6 +269,7 @@ def _resolve_inbounds(self): settings['header_type'] = '' settings['path'] = net_settings.get('serviceName', '') settings['host'] = [] + settings['multiMode'] = net_settings.get('multiMode', False) elif net == 'quic': settings['header_type'] = net_settings.get('header', {}).get('type', '') From af1593f14e8019fd3c21d4d397c81d96c7497bc7 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 16:05:35 +0330 Subject: [PATCH 09/52] fix: early data for clash --- app/subscription/clash.py | 17 ++++++++++++++++- 1 file changed, 16 insertions(+), 1 deletion(-) diff --git a/app/subscription/clash.py b/app/subscription/clash.py index 7e540d17c..ffda30796 100644 --- a/app/subscription/clash.py +++ b/app/subscription/clash.py @@ -77,6 +77,14 @@ def make_node(self, 'udp': udp } + if "?ed=" in path: + path, max_early_data = path.split("?ed=") + max_early_data, = max_early_data.split("/") + max_early_data = int(max_early_data) + early_data_header_name = "Sec-WebSocket-Protocol" + else: + max_early_data = None + if type == 'ss': # shadowsocks return node @@ -101,11 +109,18 @@ def make_node(self, net_opts['method'] = 'GET' net_opts['Host'] = host - if network == 'ws': + if network == 'ws' or network == 'httpupgrade': if path: net_opts['path'] = path if host: net_opts['headers'] = {"Host": host} + if max_early_data: + net_opts['max-early-data'] = max_early_data + net_opts['early-data-header-name'] = early_data_header_name + if network == 'httpupgrade': + net_opts['v2ray-http-upgrade'] = True + if max_early_data: + net_opts['v2ray-http-upgrade-fast-open'] = True if network == 'grpc': if path: From 17070e3b6a62ebf2338c53707cc5c8d76bfabf03 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 16:10:46 +0330 Subject: [PATCH 10/52] fix: kcp for v2ray json --- app/subscription/v2ray.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index ca6974727..ca263e9f7 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -375,7 +375,7 @@ def quic_config(path=None, host=None, header=None): return quicSettings @staticmethod - def kpc_config(path=None, host=None, header=None): + def kcp_config(path=None, host=None, header=None): kcpSettings = {} kcpSettings["header"] = {} @@ -419,7 +419,7 @@ def stream_setting_config(network=None, security=None, streamSettings["grpcSettings"] = network_setting elif network == "h2": streamSettings["httpSettings"] = network_setting - elif network == "kpc": + elif network == "kcp": streamSettings["kcpSettings"] = network_setting elif network == "tcp" and network_setting: streamSettings["tcpSettings"] = network_setting @@ -543,8 +543,8 @@ def make_stream_setting(self, network_setting = self.grpc_config(path=path, multiMode=multiMode) elif net == "h2": network_setting = self.h2_config(path=path, host=host) - elif net == "kpc": - network_setting = self.kpc_config( + elif net == "kcp": + network_setting = self.kcp_config( path=path, host=host, header=headers) elif net == "tcp": network_setting = self.tcp_http_config(path=path, host=host) From 2a68c87e8f114e1355258bf1aba86768a1e54c5e Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 21:09:12 +0330 Subject: [PATCH 11/52] fix: spiderx bug --- app/subscription/share.py | 6 +++--- app/subscription/v2ray.py | 17 +++++++++++------ app/xray/config.py | 2 +- 3 files changed, 15 insertions(+), 10 deletions(-) diff --git a/app/subscription/share.py b/app/subscription/share.py index e2a4f04be..3c9033139 100644 --- a/app/subscription/share.py +++ b/app/subscription/share.py @@ -53,7 +53,7 @@ def get_v2ray_link(remark: str, address: str, inbound: dict, settings: dict): type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), - mode=inbound.get("multiMode", False), + multiMode=inbound.get("multiMode", False), ) if inbound["protocol"] == "vless": @@ -76,7 +76,7 @@ def get_v2ray_link(remark: str, address: str, inbound: dict, settings: dict): type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), - mode=inbound.get("multiMode", False), + multiMode=inbound.get("multiMode", False), ) if inbound["protocol"] == "trojan": @@ -99,7 +99,7 @@ def get_v2ray_link(remark: str, address: str, inbound: dict, settings: dict): type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), - mode=inbound.get("multiMode", False), + multiMode=inbound.get("multiMode", False), ) if inbound["protocol"] == "shadowsocks": diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index ca263e9f7..069c4d874 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -60,7 +60,8 @@ def vmess( payload["fp"] = fp payload["pbk"] = pbk payload["sid"] = sid - payload["spx"] = spx + if spx: + payload["spx"] = spx if net == "grpc": if multiMode: @@ -134,7 +135,8 @@ def vless(cls, payload["fp"] = fp payload["pbk"] = pbk payload["sid"] = sid - payload["spx"] = spx + if spx: + payload["spx"] = spx return ( "vless://" @@ -202,7 +204,8 @@ def trojan(cls, payload["fp"] = fp payload["pbk"] = pbk payload["sid"] = sid - payload["spx"] = spx + if spx: + payload["spx"] = spx return ( "trojan://" @@ -357,7 +360,7 @@ def h2_config(path=None, host=None): return httpSettings @staticmethod - def quic_config(path=None, host=None, header=None): + def quic_config(path=None, host=None, header=None,): quicSettings = {} quicSettings["header"] = {"none"} @@ -366,11 +369,13 @@ def quic_config(path=None, host=None, header=None): else: quicSettings["key"] = "" if host: - quicSettings["security"] = [host] + quicSettings["security"] = host else: - quicSettings["security"] = "" + quicSettings["security"] = "none" if header: quicSettings["header"]["type"] = header + else: + quicSettings["header"]["type"] = "none" return quicSettings diff --git a/app/xray/config.py b/app/xray/config.py index 27f648c4a..b1a3c2081 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -229,7 +229,7 @@ def _resolve_inbounds(self): try: settings['spx'] = tls_settings.get('SpiderX') except: - pass + settings['spx'] = "" if net == 'tcp': header = net_settings.get('header', {}) From 98ee34fb61bc7f26bd25401901ea8d95ce3d0267 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 21:12:53 +0330 Subject: [PATCH 12/52] feat: add json config for v2rayn --- app/views/subscription.py | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/app/views/subscription.py b/app/views/subscription.py index 844d9535e..f8260abef 100644 --- a/app/views/subscription.py +++ b/app/views/subscription.py @@ -98,7 +98,16 @@ def get_subscription_user_info(user: UserResponse) -> dict: return Response(content=conf, media_type="application/json", headers=response_headers) else: conf = generate_subscription(user=user, config_format="v2ray", as_base64=True) + return Response(content=conf, media_type="text/plain", headers=response_headers) + + elif re.match('^v2rayN/(\d+\.\d+)', user_agent): + version_str = re.match('^v2rayN/(\d+\.\d+)', user_agent).group(1) + if LooseVersion(version_str) >= LooseVersion("6.40"): + conf = generate_subscription(user=user, config_format="v2ray-json", as_base64=False) return Response(content=conf, media_type="application/json", headers=response_headers) + else: + conf = generate_subscription(user=user, config_format="v2ray", as_base64=True) + return Response(content=conf, media_type="text/plain", headers=response_headers) else: conf = generate_subscription(user=user, config_format="v2ray", as_base64=True) From 13f33740cb2f836a69832a4ae14753b4b466c599 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Sun, 28 Apr 2024 23:21:15 +0330 Subject: [PATCH 13/52] refactor: V2rayShareLink usage --- app/subscription/share.py | 121 +++++--------------------------------- app/subscription/v2ray.py | 91 ++++++++++++++++++++++++++++ 2 files changed, 105 insertions(+), 107 deletions(-) diff --git a/app/subscription/share.py b/app/subscription/share.py index 3c9033139..10e9541ff 100644 --- a/app/subscription/share.py +++ b/app/subscription/share.py @@ -33,88 +33,11 @@ "on_hold": ONHOLD_STATUS_TEXT, } -def get_v2ray_link(remark: str, address: str, inbound: dict, settings: dict): - if inbound["protocol"] == "vmess": - return V2rayShareLink.vmess( - remark=remark, - address=address, - port=inbound["port"], - id=settings["id"], - net=inbound["network"], - tls=inbound["tls"], - sni=inbound.get("sni", ""), - fp=inbound.get("fp", ""), - alpn=inbound.get("alpn", ""), - pbk=inbound.get("pbk", ""), - sid=inbound.get("sid", ""), - spx=inbound.get("spx", ""), - host=inbound["host"], - path=inbound["path"], - type=inbound["header_type"], - ais=inbound.get("ais", ""), - fs=inbound.get("fragment_setting", ""), - multiMode=inbound.get("multiMode", False), - ) - - if inbound["protocol"] == "vless": - return V2rayShareLink.vless( - remark=remark, - address=address, - port=inbound["port"], - id=settings["id"], - flow=settings.get("flow", ""), - net=inbound["network"], - tls=inbound["tls"], - sni=inbound.get("sni", ""), - fp=inbound.get("fp", ""), - alpn=inbound.get("alpn", ""), - pbk=inbound.get("pbk", ""), - sid=inbound.get("sid", ""), - spx=inbound.get("spx", ""), - host=inbound["host"], - path=inbound["path"], - type=inbound["header_type"], - ais=inbound.get("ais", ""), - fs=inbound.get("fragment_setting", ""), - multiMode=inbound.get("multiMode", False), - ) - - if inbound["protocol"] == "trojan": - return V2rayShareLink.trojan( - remark=remark, - address=address, - port=inbound["port"], - password=settings["password"], - flow=settings.get("flow", ""), - net=inbound["network"], - tls=inbound["tls"], - sni=inbound.get("sni", ""), - fp=inbound.get("fp", ""), - alpn=inbound.get("alpn", ""), - pbk=inbound.get("pbk", ""), - sid=inbound.get("sid", ""), - spx=inbound.get("spx", ""), - host=inbound["host"], - path=inbound["path"], - type=inbound["header_type"], - ais=inbound.get("ais", ""), - fs=inbound.get("fragment_setting", ""), - multiMode=inbound.get("multiMode", False), - ) - - if inbound["protocol"] == "shadowsocks": - return V2rayShareLink.shadowsocks( - remark=remark, - address=address, - port=inbound["port"], - password=settings["password"], - method=settings["method"], - ) - def generate_v2ray_links(proxies: dict, inbounds: dict, extra_data: dict) -> list: format_variables = setup_format_variables(extra_data) - return process_inbounds_and_tags(inbounds, proxies, format_variables, mode="v2ray") + conf = V2rayShareLink() + return process_inbounds_and_tags(inbounds, proxies, format_variables, conf=conf) def generate_clash_subscription( @@ -127,7 +50,7 @@ def generate_clash_subscription( format_variables = setup_format_variables(extra_data) return process_inbounds_and_tags( - inbounds, proxies, format_variables, mode="clash", conf=conf + inbounds, proxies, format_variables, conf=conf ) @@ -138,7 +61,7 @@ def generate_singbox_subscription( format_variables = setup_format_variables(extra_data) return process_inbounds_and_tags( - inbounds, proxies, format_variables, mode="sing-box", conf=conf + inbounds, proxies, format_variables, conf=conf ) @@ -153,7 +76,7 @@ def generate_outline_subscription( format_variables = setup_format_variables(extra_data) return process_inbounds_and_tags( - inbounds, proxies, format_variables, mode="outline", conf=conf + inbounds, proxies, format_variables, conf=conf ) @@ -164,7 +87,7 @@ def generate_v2ray_json_subscription( format_variables = setup_format_variables(extra_data) return process_inbounds_and_tags( - inbounds, proxies, format_variables, mode="v2ray-json", conf=conf + inbounds, proxies, format_variables, conf=conf ) @@ -298,10 +221,8 @@ def process_inbounds_and_tags( inbounds: dict, proxies: dict, format_variables: dict, - mode: str = "v2ray", conf=None, ) -> Union[List, str]: - results = [] _inbounds = [] for protocol, tags in inbounds.items(): @@ -359,28 +280,14 @@ def process_inbounds_and_tags( } ) - if mode == "v2ray": - results.append( - get_v2ray_link( - remark=host["remark"].format_map(format_variables), - address=address.format_map( - format_variables), - inbound=host_inbound, - settings=settings.dict(no_obj=True), - ) - ) - elif mode in ["clash", "sing-box", "outline", "v2ray-json"]: - conf.add( - remark=host["remark"].format_map(format_variables), - address=address.format_map(format_variables), - inbound=host_inbound, - settings=settings.dict(no_obj=True), - ) - - if mode in ["clash", "sing-box", "outline", "v2ray-json"]: - return conf.render() - - return results + conf.add( + remark=host["remark"].format_map(format_variables), + address=address.format_map(format_variables), + inbound=host_inbound, + settings=settings.dict(no_obj=True), + ) + + return conf.render() def encode_title(text: str) -> str: diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index 069c4d874..586473058 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -10,6 +10,97 @@ class V2rayShareLink(str): + def __init__(self): + self.links = [] + + def add_link(self, link): + self.links.append(link) + + def render(self): + return self.links + + def add(self, remark: str, address: str, inbound: dict, settings: dict): + + if inbound["protocol"] == "vmess": + link = self.vmess( + remark=remark, + address=address, + port=inbound["port"], + id=settings["id"], + net=inbound["network"], + tls=inbound["tls"], + sni=inbound.get("sni", ""), + fp=inbound.get("fp", ""), + alpn=inbound.get("alpn", ""), + pbk=inbound.get("pbk", ""), + sid=inbound.get("sid", ""), + spx=inbound.get("spx", ""), + host=inbound["host"], + path=inbound["path"], + type=inbound["header_type"], + ais=inbound.get("ais", ""), + fs=inbound.get("fragment_setting", ""), + multiMode=inbound.get("multiMode", False), + ) + + elif inbound["protocol"] == "vless": + link = self.vless( + remark=remark, + address=address, + port=inbound["port"], + id=settings["id"], + flow=settings.get("flow", ""), + net=inbound["network"], + tls=inbound["tls"], + sni=inbound.get("sni", ""), + fp=inbound.get("fp", ""), + alpn=inbound.get("alpn", ""), + pbk=inbound.get("pbk", ""), + sid=inbound.get("sid", ""), + spx=inbound.get("spx", ""), + host=inbound["host"], + path=inbound["path"], + type=inbound["header_type"], + ais=inbound.get("ais", ""), + fs=inbound.get("fragment_setting", ""), + multiMode=inbound.get("multiMode", False), + ) + + elif inbound["protocol"] == "trojan": + link = self.trojan( + remark=remark, + address=address, + port=inbound["port"], + password=settings["password"], + flow=settings.get("flow", ""), + net=inbound["network"], + tls=inbound["tls"], + sni=inbound.get("sni", ""), + fp=inbound.get("fp", ""), + alpn=inbound.get("alpn", ""), + pbk=inbound.get("pbk", ""), + sid=inbound.get("sid", ""), + spx=inbound.get("spx", ""), + host=inbound["host"], + path=inbound["path"], + type=inbound["header_type"], + ais=inbound.get("ais", ""), + fs=inbound.get("fragment_setting", ""), + multiMode=inbound.get("multiMode", False), + ) + + elif inbound["protocol"] == "shadowsocks": + link = self.shadowsocks( + remark=remark, + address=address, + port=inbound["port"], + password=settings["password"], + method=settings["method"], + ) + + self.add_link(link=link) + + @classmethod def vmess( cls, From 41ae408e843a95cad93c7ce5d5178b6b1aa02a72 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Mon, 29 Apr 2024 00:04:52 +0330 Subject: [PATCH 14/52] fix: status change for not on hold users --- app/jobs/review_users.py | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/app/jobs/review_users.py b/app/jobs/review_users.py index 7da8d90fa..41aa96947 100644 --- a/app/jobs/review_users.py +++ b/app/jobs/review_users.py @@ -55,10 +55,8 @@ def review(): xray.operations.remove_user(user) update_user_status(db, user, status) - bg.add_task( - report.status_change, username=user.username, status=status, - user=UserResponse.from_orm(user), user_admin=user.admin - ) + report.status_change(username=user.username, status=status, + user=UserResponse.from_orm(user), user_admin=user.admin) logger.info(f"User \"{user.username}\" status changed to {status}") From dae8eb8130b8e5bf3c6ed92918fbd2c64cd05202 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Tue, 30 Apr 2024 15:26:59 +0330 Subject: [PATCH 15/52] feat: support gRPC Custom Path --- app/subscription/clash.py | 3 +++ app/subscription/funcs.py | 9 +++++++++ app/subscription/singbox.py | 6 +++++- app/subscription/v2ray.py | 12 ++++++++---- app/xray/config.py | 4 +++- 5 files changed, 28 insertions(+), 6 deletions(-) create mode 100644 app/subscription/funcs.py diff --git a/app/subscription/clash.py b/app/subscription/clash.py index ffda30796..947a52271 100644 --- a/app/subscription/clash.py +++ b/app/subscription/clash.py @@ -1,6 +1,7 @@ import yaml import json from app.templates import render_template +from app.subscription.funcs import grpc_correct_path from config import CLASH_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE @@ -61,6 +62,8 @@ def make_node(self, ais: bool = '', mux_enable : bool = False): + path = grpc_correct_path(path=path, word="customTunMulti") + if type == 'shadowsocks': type = 'ss' if network == 'tcp' and headers == 'http': diff --git a/app/subscription/funcs.py b/app/subscription/funcs.py new file mode 100644 index 000000000..27aa2ee50 --- /dev/null +++ b/app/subscription/funcs.py @@ -0,0 +1,9 @@ +import re +from typing import List + +def grpc_correct_path(path: str, word: str): + + pattern = r'\b(?:' + re.escape(word) + r'|[|])\b' + path = re.sub(pattern, '', path) + + return path \ No newline at end of file diff --git a/app/subscription/singbox.py b/app/subscription/singbox.py index fb38f8e0e..819ffab93 100644 --- a/app/subscription/singbox.py +++ b/app/subscription/singbox.py @@ -1,5 +1,6 @@ import json from app.templates import render_template +from app.subscription.funcs import grpc_correct_path from config import SINGBOX_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE @@ -193,6 +194,9 @@ def make_outbound(self, return config def add(self, remark: str, address: str, inbound: dict, settings: dict): + + path = grpc_correct_path(path=inbound["path"], word="customTunMulti") + outbound = self.make_outbound( remark=remark, type=inbound['protocol'], @@ -203,7 +207,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): flow=settings.get('flow', ''), sni=inbound['sni'], host=inbound['host'], - path=inbound['path'], + path=path, alpn=inbound.get('alpn', ''), fp=inbound.get('fp', ''), pbk=inbound.get('pbk', ''), diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index 586473058..d214ccdd4 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -4,6 +4,7 @@ from typing import Union from uuid import UUID +from app.subscription.funcs import grpc_correct_path from app.templates import render_template from config import (MUX_TEMPLATE, V2RAY_SUBSCRIPTION_TEMPLATE) @@ -21,6 +22,8 @@ def render(self): def add(self, remark: str, address: str, inbound: dict, settings: dict): + path = grpc_correct_path(path=inbound["path"], word="customTun") + if inbound["protocol"] == "vmess": link = self.vmess( remark=remark, @@ -36,7 +39,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): sid=inbound.get("sid", ""), spx=inbound.get("spx", ""), host=inbound["host"], - path=inbound["path"], + path=path, type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), @@ -59,7 +62,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): sid=inbound.get("sid", ""), spx=inbound.get("spx", ""), host=inbound["host"], - path=inbound["path"], + path=path, type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), @@ -82,7 +85,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): sid=inbound.get("sid", ""), spx=inbound.get("spx", ""), host=inbound["host"], - path=inbound["path"], + path=path, type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), @@ -680,6 +683,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): tls = (inbound['tls']) headers = inbound['header_type'] fragment = inbound['fragment_setting'] + path = grpc_correct_path(path=inbound["path"], word="customTun") outbound = { "tag": remark, @@ -735,7 +739,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): tls=tls, sni=inbound['sni'], host=inbound['host'], - path=inbound['path'], + path=path, alpn=inbound.get('alpn', ''), fp=inbound.get('fp', ''), pbk=inbound.get('pbk', ''), diff --git a/app/xray/config.py b/app/xray/config.py index b1a3c2081..f74ea2399 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -265,11 +265,13 @@ def _resolve_inbounds(self): if isinstance(host, str): settings['host'] = [host] - elif net == 'grpc': + elif net == 'grpc' or net == 'gun': settings['header_type'] = '' settings['path'] = net_settings.get('serviceName', '') settings['host'] = [] settings['multiMode'] = net_settings.get('multiMode', False) + if 'customTunMulti' in settings['path']: + settings['multiMode'] = True elif net == 'quic': settings['header_type'] = net_settings.get('header', {}).get('type', '') From fb2f01e34af580156661db96cd2c41fc15deacdf Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Tue, 30 Apr 2024 15:27:56 +0330 Subject: [PATCH 16/52] fix: remove level and set it to default --- app/subscription/v2ray.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index d214ccdd4..525e28ad9 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -579,7 +579,6 @@ def trojan_config(address=None, port=None, password=None, method="chacha20"): servers["email"] = "https://gozargah.github.io/marzban/" servers["method"] = method servers["ota"] = False - servers["level"] = 1 settings["servers"] = [servers] @@ -597,7 +596,6 @@ def shadowsocks_config(address=None, port=None, password=None, method=None): servers["email"] = "https://gozargah.github.io/marzban/" servers["method"] = method servers["uot"] = False - servers["level"] = 1 settings["servers"] = [servers] From daa609cd479cf140522b2bb188cf5dcaf01a7da1 Mon Sep 17 00:00:00 2001 From: Stanislav Date: Sun, 19 May 2024 21:48:33 +0300 Subject: [PATCH 17/52] fix pg migrations --- .../migrations/versions/51e941ed9018_deactive_user_status.py | 3 +++ .../versions/7cbe9d91ac11_proxyhost_security_added.py | 3 +++ .../versions/c106bb40c861_alpn_fingerprint_hosts.py | 5 +++++ .../d02dcfbf1517_add_userusageresetlogs_model_and_data_.py | 3 +++ 4 files changed, 14 insertions(+) diff --git a/app/db/migrations/versions/51e941ed9018_deactive_user_status.py b/app/db/migrations/versions/51e941ed9018_deactive_user_status.py index cbd4ed9a5..1a5193d2b 100644 --- a/app/db/migrations/versions/51e941ed9018_deactive_user_status.py +++ b/app/db/migrations/versions/51e941ed9018_deactive_user_status.py @@ -41,6 +41,9 @@ def upgrade(): + status_enum = sa.Enum('active', 'limited', 'expired', name='status') + status_enum.create(op.get_bind()) + # temp type to use instead of old one temp_type.create(op.get_bind(), checkfirst=False) diff --git a/app/db/migrations/versions/7cbe9d91ac11_proxyhost_security_added.py b/app/db/migrations/versions/7cbe9d91ac11_proxyhost_security_added.py index 18b1c3e0e..02b6e5000 100644 --- a/app/db/migrations/versions/7cbe9d91ac11_proxyhost_security_added.py +++ b/app/db/migrations/versions/7cbe9d91ac11_proxyhost_security_added.py @@ -17,6 +17,9 @@ def upgrade() -> None: + proxyhostsecurity_enum = sa.Enum('inbound_default', 'none', 'tls', name='proxyhostsecurity') + proxyhostsecurity_enum.create(op.get_bind()) + # ### commands auto generated by Alembic - please adjust! ### op.add_column( "hosts", diff --git a/app/db/migrations/versions/c106bb40c861_alpn_fingerprint_hosts.py b/app/db/migrations/versions/c106bb40c861_alpn_fingerprint_hosts.py index 55691911c..e713922b0 100644 --- a/app/db/migrations/versions/c106bb40c861_alpn_fingerprint_hosts.py +++ b/app/db/migrations/versions/c106bb40c861_alpn_fingerprint_hosts.py @@ -17,6 +17,11 @@ def upgrade() -> None: + proxyhostalpn_enum = sa.Enum('none', 'h2', 'http/1.1', 'h2,http/1.1', name='proxyhostalpn') + proxyhostalpn_enum.create(op.get_bind(), checkfirst=True) + + proxyhostfingerprint_enum = sa.Enum('none', 'chrome', 'firefox', 'safari', 'ios', 'android', 'edge', '360', 'qq', 'random', 'randomized', name='proxyhostfingerprint') + proxyhostfingerprint_enum.create(op.get_bind(), checkfirst=True) # ### commands auto generated by Alembic - please adjust! ### op.add_column('hosts', sa.Column('alpn', sa.Enum('none', 'h2', 'http/1.1', 'h2,http/1.1', name='proxyhostalpn'), server_default='none', nullable=False)) op.add_column('hosts', sa.Column('fingerprint', sa.Enum('none', 'chrome', 'firefox', 'safari', 'ios', 'android', 'edge', '360', 'qq', 'random', 'randomized', name='proxyhostfingerprint'), server_default='none', nullable=False)) diff --git a/app/db/migrations/versions/d02dcfbf1517_add_userusageresetlogs_model_and_data_.py b/app/db/migrations/versions/d02dcfbf1517_add_userusageresetlogs_model_and_data_.py index c1bf6670c..5a204a730 100644 --- a/app/db/migrations/versions/d02dcfbf1517_add_userusageresetlogs_model_and_data_.py +++ b/app/db/migrations/versions/d02dcfbf1517_add_userusageresetlogs_model_and_data_.py @@ -17,6 +17,9 @@ def upgrade() -> None: + userdatalimitresetstrategy = sa.Enum('no_reset', 'day', 'week', 'month', 'year', name='userdatalimitresetstrategy') + userdatalimitresetstrategy.create(op.get_bind()) + # ### commands auto generated by Alembic - please adjust! ### op.create_table('user_usage_logs', sa.Column('id', sa.Integer(), nullable=False), From 7993196e4cac42ebfda76ea352d9ac7fc0f0a6d8 Mon Sep 17 00:00:00 2001 From: "Mahan MI(PC)" Date: Wed, 29 May 2024 21:54:16 +0330 Subject: [PATCH 18/52] feat: :bug: Disabled Clash Meta Flow for non-TLS TCP & KCP configs non-TLS configs doesn't support flow and the config won't work if flow has been set so , i disabled it for non-TLS configs in clash-meta file (it was disabled for other clients) --- app/subscription/clash.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/subscription/clash.py b/app/subscription/clash.py index e999055b7..2109eb7f9 100644 --- a/app/subscription/clash.py +++ b/app/subscription/clash.py @@ -244,7 +244,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): if inbound['protocol'] == 'vless': node['uuid'] = settings['id'] - if inbound['network'] in ('tcp', 'kcp') and inbound['header_type'] != 'http': + if inbound['network'] in ('tcp', 'kcp') and inbound['header_type'] != 'http' and inbound['tls'] != 'none': node['flow'] = settings.get('flow', '') self.data['proxies'].append(node) From 967fdf0020c850a7bfa9e676be8b590b9a1060c5 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Thu, 30 May 2024 15:13:45 +0330 Subject: [PATCH 19/52] fix: grpc encode --- app/subscription/clash.py | 6 ++++-- app/subscription/funcs.py | 23 +++++++++++++++------ app/subscription/singbox.py | 10 +++++++--- app/subscription/v2ray.py | 40 +++++++++++++++++++++++++++---------- app/xray/config.py | 2 -- 5 files changed, 58 insertions(+), 23 deletions(-) diff --git a/app/subscription/clash.py b/app/subscription/clash.py index 947a52271..afa66f385 100644 --- a/app/subscription/clash.py +++ b/app/subscription/clash.py @@ -1,7 +1,7 @@ import yaml import json from app.templates import render_template -from app.subscription.funcs import grpc_correct_path +from app.subscription.funcs import get_grpc_gun from config import CLASH_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE @@ -62,7 +62,9 @@ def make_node(self, ais: bool = '', mux_enable : bool = False): - path = grpc_correct_path(path=path, word="customTunMulti") + + if network in ["grpc", "gun"]: + path = get_grpc_gun(path) if type == 'shadowsocks': type = 'ss' diff --git a/app/subscription/funcs.py b/app/subscription/funcs.py index 27aa2ee50..9fcc3d0de 100644 --- a/app/subscription/funcs.py +++ b/app/subscription/funcs.py @@ -1,9 +1,20 @@ -import re -from typing import List +def get_grpc_gun(path: str) -> str: + if not path.startswith("/"): + return path -def grpc_correct_path(path: str, word: str): + servicename = path.rsplit("/", 1)[0] + streamname = path.rsplit("/", 1)[1].split("|")[0] + + if streamname == "Tun": + return servicename[1:] + + return "%s%s%s" % (servicename, "/", streamname) - pattern = r'\b(?:' + re.escape(word) + r'|[|])\b' - path = re.sub(pattern, '', path) +def get_grpc_multi(path: str) -> str: + if not path.startswith("/"): + return path + + servicename = path.rsplit("/", 1)[0] + streamname = path.rsplit("/", 1)[1].split("|")[1] - return path \ No newline at end of file + return "%s%s%s" % (servicename, "/", streamname) \ No newline at end of file diff --git a/app/subscription/singbox.py b/app/subscription/singbox.py index 819ffab93..163cf8ed5 100644 --- a/app/subscription/singbox.py +++ b/app/subscription/singbox.py @@ -1,6 +1,6 @@ import json from app.templates import render_template -from app.subscription.funcs import grpc_correct_path +from app.subscription.funcs import get_grpc_gun from config import SINGBOX_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE @@ -195,14 +195,18 @@ def make_outbound(self, def add(self, remark: str, address: str, inbound: dict, settings: dict): - path = grpc_correct_path(path=inbound["path"], word="customTunMulti") + net = inbound["network"] + path = inbound["path"] + + if net in ["grpc", "gun"]: + path = get_grpc_gun(path) outbound = self.make_outbound( remark=remark, type=inbound['protocol'], address=address, port=inbound['port'], - net=inbound['network'], + net=net, tls=(inbound['tls']), flow=settings.get('flow', ''), sni=inbound['sni'], diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index 525e28ad9..199e2165b 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -3,8 +3,9 @@ import urllib.parse as urlparse from typing import Union from uuid import UUID +from urllib.parse import quote -from app.subscription.funcs import grpc_correct_path +from app.subscription.funcs import get_grpc_gun, get_grpc_multi from app.templates import render_template from config import (MUX_TEMPLATE, V2RAY_SUBSCRIPTION_TEMPLATE) @@ -21,8 +22,20 @@ def render(self): return self.links def add(self, remark: str, address: str, inbound: dict, settings: dict): + net = inbound["network"] + multi_mode = inbound.get("multiMode", False) + old_path: str = inbound["path"] - path = grpc_correct_path(path=inbound["path"], word="customTun") + if net in ["grpc", "gun"]: + if multi_mode: + path = get_grpc_multi(old_path) + else: + path = get_grpc_gun(old_path) + if old_path.startswith("/"): + path = quote(path, safe="-_.!~*'()") + + else: + path = old_path if inbound["protocol"] == "vmess": link = self.vmess( @@ -30,7 +43,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): address=address, port=inbound["port"], id=settings["id"], - net=inbound["network"], + net=net, tls=inbound["tls"], sni=inbound.get("sni", ""), fp=inbound.get("fp", ""), @@ -43,7 +56,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), - multiMode=inbound.get("multiMode", False), + multiMode=multi_mode, ) elif inbound["protocol"] == "vless": @@ -53,7 +66,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): port=inbound["port"], id=settings["id"], flow=settings.get("flow", ""), - net=inbound["network"], + net=net, tls=inbound["tls"], sni=inbound.get("sni", ""), fp=inbound.get("fp", ""), @@ -66,7 +79,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), - multiMode=inbound.get("multiMode", False), + multiMode=multi_mode, ) elif inbound["protocol"] == "trojan": @@ -76,7 +89,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): port=inbound["port"], password=settings["password"], flow=settings.get("flow", ""), - net=inbound["network"], + net=net, tls=inbound["tls"], sni=inbound.get("sni", ""), fp=inbound.get("fp", ""), @@ -89,7 +102,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): type=inbound["header_type"], ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), - multiMode=inbound.get("multiMode", False), + multiMode=multi_mode, ) elif inbound["protocol"] == "shadowsocks": @@ -681,7 +694,14 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): tls = (inbound['tls']) headers = inbound['header_type'] fragment = inbound['fragment_setting'] - path = grpc_correct_path(path=inbound["path"], word="customTun") + path = inbound["path"] + multi_mode = inbound.get("multiMode", False) + + if net in ["grpc", "gun"]: + if multi_mode: + path = get_grpc_multi(path) + else: + path = get_grpc_gun(path) outbound = { "tag": remark, @@ -746,7 +766,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): headers=headers, ais=inbound.get('ais', ''), dialer_proxy=dialer_proxy, - multiMode=inbound.get('multiMode',False), + multiMode=multi_mode, ) mux_json = json.loads(self.mux_template) diff --git a/app/xray/config.py b/app/xray/config.py index f74ea2399..6cb82b616 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -270,8 +270,6 @@ def _resolve_inbounds(self): settings['path'] = net_settings.get('serviceName', '') settings['host'] = [] settings['multiMode'] = net_settings.get('multiMode', False) - if 'customTunMulti' in settings['path']: - settings['multiMode'] = True elif net == 'quic': settings['header_type'] = net_settings.get('header', {}).get('type', '') From 3dfdb1f8cb7730dd6498180b0b1860cd8d2e27bd Mon Sep 17 00:00:00 2001 From: WhyMan1 <154282687+WhyMan1@users.noreply.github.com> Date: Fri, 7 Jun 2024 04:14:28 +0330 Subject: [PATCH 20/52] Fix: Httpupgrade host --- app/subscription/singbox.py | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/app/subscription/singbox.py b/app/subscription/singbox.py index 01e14cabd..9f6027fce 100644 --- a/app/subscription/singbox.py +++ b/app/subscription/singbox.py @@ -114,11 +114,9 @@ def transport_config(transport_type='', transport_config['permit_without_stream'] = permit_without_stream elif transport_type == "httpupgrade": - transport_config['host'] = "" + transport_config['host'] = host if path: transport_config['path'] = path - if host: - transport_config['headers'] = {'Host': host} return transport_config From c585b050c14a20131fc4ef658ba2fcbe07cd5872 Mon Sep 17 00:00:00 2001 From: SaintShit Date: Fri, 28 Jun 2024 01:41:12 +0330 Subject: [PATCH 21/52] feat: custom config for streisand --- .env.example | 1 + app/views/subscription.py | 11 ++++++++++- config.py | 1 + 3 files changed, 12 insertions(+), 1 deletion(-) diff --git a/.env.example b/.env.example index 68c958e4f..dfb3faf23 100644 --- a/.env.example +++ b/.env.example @@ -43,6 +43,7 @@ UVICORN_PORT = 8000 ## If USE_CUSTOM_JSON_DEFAULT is set True, all following programs will use the JSON config # USE_CUSTOM_JSON_FOR_V2RAYN=False # USE_CUSTOM_JSON_FOR_V2RAYNG=True +# USE_CUSTOM_JSON_FOR_STREISAND=False ## Set headers for subscription # SUB_PROFILE_TITLE = "Susbcription" diff --git a/app/views/subscription.py b/app/views/subscription.py index 31e021fdc..f39e28372 100644 --- a/app/views/subscription.py +++ b/app/views/subscription.py @@ -19,7 +19,8 @@ XRAY_SUBSCRIPTION_PATH, USE_CUSTOM_JSON_DEFAULT, USE_CUSTOM_JSON_FOR_V2RAYN, - USE_CUSTOM_JSON_FOR_V2RAYNG + USE_CUSTOM_JSON_FOR_V2RAYNG, + USE_CUSTOM_JSON_FOR_STREISAND ) @@ -114,6 +115,14 @@ def get_subscription_user_info(user: UserResponse) -> dict: conf = generate_subscription(user=user, config_format="v2ray", as_base64=True) return Response(content=conf, media_type="text/plain", headers=response_headers) + elif re.match('^[Ss]treisand', user_agent): + if USE_CUSTOM_JSON_DEFAULT or USE_CUSTOM_JSON_FOR_STREISAND: + conf = generate_subscription(user=user, config_format="v2ray-json", as_base64=False) + return Response(content=conf, media_type="application/json", headers=response_headers) + else: + conf = generate_subscription(user=user, config_format="v2ray", as_base64=True) + return Response(content=conf, media_type="text/plain", headers=response_headers) + else: conf = generate_subscription(user=user, config_format="v2ray", as_base64=True) return Response(content=conf, media_type="text/plain", headers=response_headers) diff --git a/config.py b/config.py index a75cb2301..fa22f994b 100755 --- a/config.py +++ b/config.py @@ -54,6 +54,7 @@ USE_CUSTOM_JSON_DEFAULT = config("USE_CUSTOM_JSON_DEFAULT", default=False, cast=bool) USE_CUSTOM_JSON_FOR_V2RAYN = config("USE_CUSTOM_JSON_FOR_V2RAYN", default=False, cast=bool) USE_CUSTOM_JSON_FOR_V2RAYNG = config("USE_CUSTOM_JSON_FOR_V2RAYNG", default=False, cast=bool) +USE_CUSTOM_JSON_FOR_STREISAND = config("USE_CUSTOM_JSON_FOR_STREISAND", default=False, cast=bool) ACTIVE_STATUS_TEXT = config("ACTIVE_STATUS_TEXT", default="Active") EXPIRED_STATUS_TEXT = config("EXPIRED_STATUS_TEXT", default="Expired") From 6f3f44bf1947af683e07dc721fb5aff369923585 Mon Sep 17 00:00:00 2001 From: oXIIIo <139705681+oXIIIo@users.noreply.github.com> Date: Sun, 30 Jun 2024 22:29:12 +0330 Subject: [PATCH 22/52] fix: add support for FlClash client app and resolve issue #1052 --- app/views/subscription.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/views/subscription.py b/app/views/subscription.py index f39e28372..c5af07f03 100644 --- a/app/views/subscription.py +++ b/app/views/subscription.py @@ -79,7 +79,7 @@ def get_subscription_user_info(user: UserResponse) -> dict: crud.update_user_sub(db, dbuser, user_agent) - if re.match('^([Cc]lash-verge|[Cc]lash-?[Mm]eta)', user_agent): + if re.match('^([Cc]lash-verge|[Cc]lash[-\.]?[Mm]eta)', user_agent): conf = generate_subscription(user=user, config_format="clash-meta", as_base64=False) return Response(content=conf, media_type="text/yaml", headers=response_headers) From 4e70e28352d8e52c097102b526cd0d78412500a8 Mon Sep 17 00:00:00 2001 From: fodhelper <161948085+fodhelper@users.noreply.github.com> Date: Mon, 1 Jul 2024 12:41:51 +0100 Subject: [PATCH 23/52] v2ray json only for 1.8.18 and newer versions --- app/views/subscription.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/views/subscription.py b/app/views/subscription.py index f39e28372..d587d7060 100644 --- a/app/views/subscription.py +++ b/app/views/subscription.py @@ -107,7 +107,7 @@ def get_subscription_user_info(user: UserResponse) -> dict: elif re.match('^v2rayNG/(\d+\.\d+\.\d+)', user_agent): version_str = re.match('^v2rayNG/(\d+\.\d+\.\d+)', user_agent).group(1) - if LooseVersion(version_str) >= LooseVersion("1.8.16") and \ + if LooseVersion(version_str) >= LooseVersion("1.8.18") and \ (USE_CUSTOM_JSON_DEFAULT or USE_CUSTOM_JSON_FOR_V2RAYNG): conf = generate_subscription(user=user, config_format="v2ray-json", as_base64=False) return Response(content=conf, media_type="application/json", headers=response_headers) From b2851335e32625624e9e966d756c2c47385e465c Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Mon, 1 Jul 2024 17:56:41 +0330 Subject: [PATCH 24/52] Merge branch 'dev' into 1.8.11 --- app/dashboard/build/locales/en.json | 1 + app/dashboard/build/locales/fa.json | 1 + app/dashboard/build/locales/ru.json | 1 + app/dashboard/build/locales/zh.json | 1 + app/dashboard/public/locales/en.json | 1 + app/dashboard/public/locales/fa.json | 1 + app/dashboard/public/locales/ru.json | 1 + app/dashboard/public/locales/zh.json | 1 + app/dashboard/src/components/HostsDialog.tsx | 24 ++++ app/db/crud.py | 1 + ...92220c0d0_add_support_random_user_agent.py | 28 +++++ app/db/models.py | 1 + app/models/proxy.py | 1 + app/subscription/share.py | 3 +- app/subscription/singbox.py | 35 +++++- app/subscription/v2ray.py | 49 ++++++--- app/templates/user_agent/default.json | 104 ++++++++++++++++++ app/xray/__init__.py | 3 +- config.py | 2 + 19 files changed, 236 insertions(+), 23 deletions(-) create mode 100644 app/db/migrations/versions/31f92220c0d0_add_support_random_user_agent.py create mode 100644 app/templates/user_agent/default.json diff --git a/app/dashboard/build/locales/en.json b/app/dashboard/build/locales/en.json index 09f49343f..a81f47b59 100644 --- a/app/dashboard/build/locales/en.json +++ b/app/dashboard/build/locales/en.json @@ -105,6 +105,7 @@ "hostsDialog.host.wildcard": "Use * to generate a random string (works for wildcard domain names)", "hostsDialog.sockopt": "Sockopt", "hostsDialog.muxEnable": "Enable MUX", + "hostsDialog.randomUserAgent":"Use random user agent", "hostsDialog.allowinsecure": "Allow Insecure", "hostsDialog.fragment": "Fragment pattern", "hostsDialog.fragment.info": "Correct pattern: length,interval,packets", diff --git a/app/dashboard/build/locales/fa.json b/app/dashboard/build/locales/fa.json index 57e79ace5..0b0fbd88a 100644 --- a/app/dashboard/build/locales/fa.json +++ b/app/dashboard/build/locales/fa.json @@ -106,6 +106,7 @@ "hostsDialog.host.wildcard": "از * برای ساخت عبارت تصادفی استفاده کنید (برای نام‌های wildcard کار می‌کند)", "hostsDialog.sockopt": "Sockopt", "hostsDialog.muxEnable": "فعالسازی MUX", + "hostsDialog.randomUserAgent":"استفاده از User Agent تصادفی", "hostsDialog.allowinsecure": "Allow Insecure", "hostsDialog.fragment": "الگو فرگمنت", "hostsDialog.fragment.info": "length,interval,packet (e.g. 10-100,100-200,tlshello)", diff --git a/app/dashboard/build/locales/ru.json b/app/dashboard/build/locales/ru.json index 7d3409aa0..6e52cec13 100644 --- a/app/dashboard/build/locales/ru.json +++ b/app/dashboard/build/locales/ru.json @@ -103,6 +103,7 @@ "hostsDialog.host.wildcard": "Используйте *, чтобы сгенерировать случайную строку (работает для wildcard доменов)", "hostsDialog.sockopt": "Sockopt", "hostsDialog.muxEnable": "Давать возможность MUX", + "hostsDialog.randomUserAgent":"Use random user agent", "hostsDialog.allowinsecure": "Allow Insecure", "hostsDialog.fragment": "Шаблон фрагмента", "hostsDialog.fragment.info": "length,interval,packet (e.g. 10-100,100-200,tlshello)", diff --git a/app/dashboard/build/locales/zh.json b/app/dashboard/build/locales/zh.json index f9817e286..65789ad45 100644 --- a/app/dashboard/build/locales/zh.json +++ b/app/dashboard/build/locales/zh.json @@ -97,6 +97,7 @@ "hostsDialog.fingerprint": "指纹", "hostsDialog.sockopt": "Sockopt", "hostsDialog.muxEnable": "使能够 MUX", + "hostsDialog.randomUserAgent":"Use random user agent", "hostsDialog.allowinsecure": "Allow Insecure", "hostsDialog.fragment": "碎片图案", "hostsDialog.fragment.info": "length,interval,packet (e.g. 10-100,100-200,tlshello)", diff --git a/app/dashboard/public/locales/en.json b/app/dashboard/public/locales/en.json index 09f49343f..a81f47b59 100644 --- a/app/dashboard/public/locales/en.json +++ b/app/dashboard/public/locales/en.json @@ -105,6 +105,7 @@ "hostsDialog.host.wildcard": "Use * to generate a random string (works for wildcard domain names)", "hostsDialog.sockopt": "Sockopt", "hostsDialog.muxEnable": "Enable MUX", + "hostsDialog.randomUserAgent":"Use random user agent", "hostsDialog.allowinsecure": "Allow Insecure", "hostsDialog.fragment": "Fragment pattern", "hostsDialog.fragment.info": "Correct pattern: length,interval,packets", diff --git a/app/dashboard/public/locales/fa.json b/app/dashboard/public/locales/fa.json index 57e79ace5..0b0fbd88a 100644 --- a/app/dashboard/public/locales/fa.json +++ b/app/dashboard/public/locales/fa.json @@ -106,6 +106,7 @@ "hostsDialog.host.wildcard": "از * برای ساخت عبارت تصادفی استفاده کنید (برای نام‌های wildcard کار می‌کند)", "hostsDialog.sockopt": "Sockopt", "hostsDialog.muxEnable": "فعالسازی MUX", + "hostsDialog.randomUserAgent":"استفاده از User Agent تصادفی", "hostsDialog.allowinsecure": "Allow Insecure", "hostsDialog.fragment": "الگو فرگمنت", "hostsDialog.fragment.info": "length,interval,packet (e.g. 10-100,100-200,tlshello)", diff --git a/app/dashboard/public/locales/ru.json b/app/dashboard/public/locales/ru.json index 7d3409aa0..6e52cec13 100644 --- a/app/dashboard/public/locales/ru.json +++ b/app/dashboard/public/locales/ru.json @@ -103,6 +103,7 @@ "hostsDialog.host.wildcard": "Используйте *, чтобы сгенерировать случайную строку (работает для wildcard доменов)", "hostsDialog.sockopt": "Sockopt", "hostsDialog.muxEnable": "Давать возможность MUX", + "hostsDialog.randomUserAgent":"Use random user agent", "hostsDialog.allowinsecure": "Allow Insecure", "hostsDialog.fragment": "Шаблон фрагмента", "hostsDialog.fragment.info": "length,interval,packet (e.g. 10-100,100-200,tlshello)", diff --git a/app/dashboard/public/locales/zh.json b/app/dashboard/public/locales/zh.json index f9817e286..65789ad45 100644 --- a/app/dashboard/public/locales/zh.json +++ b/app/dashboard/public/locales/zh.json @@ -97,6 +97,7 @@ "hostsDialog.fingerprint": "指纹", "hostsDialog.sockopt": "Sockopt", "hostsDialog.muxEnable": "使能够 MUX", + "hostsDialog.randomUserAgent":"Use random user agent", "hostsDialog.allowinsecure": "Allow Insecure", "hostsDialog.fragment": "碎片图案", "hostsDialog.fragment.info": "length,interval,packet (e.g. 10-100,100-200,tlshello)", diff --git a/app/dashboard/src/components/HostsDialog.tsx b/app/dashboard/src/components/HostsDialog.tsx index 9eb8efad3..89d79d058 100644 --- a/app/dashboard/src/components/HostsDialog.tsx +++ b/app/dashboard/src/components/HostsDialog.tsx @@ -119,6 +119,7 @@ const hostsSchema = z.record( allowinsecure: z.boolean().nullable().default(false), is_disabled: z.boolean().default(true), fragment_setting: z.string().nullable(), + random_user_agent: z.boolean().default(false), security: z.string(), alpn: z.string(), fingerprint: z.string(), @@ -175,6 +176,7 @@ const AccordionInbound: FC = ({ allowinsecure: false, is_disabled: false, fragment_setting: "", + random_user_agent: false, security: "inbound_default", alpn: "", fingerprint: "", @@ -920,6 +922,28 @@ const AccordionInbound: FC = ({ )} + + + {t("hostsDialog.randomUserAgent")} + + {accordionErrors && + accordionErrors[index]?.random_user_agent && ( + + {accordionErrors[index]?.random_user_agent?.message} + + )} + diff --git a/app/db/crud.py b/app/db/crud.py index cb2edb1d7..1affebb46 100644 --- a/app/db/crud.py +++ b/app/db/crud.py @@ -85,6 +85,7 @@ def update_hosts(db: Session, inbound_tag: str, modified_hosts: List[ProxyHostMo is_disabled=host.is_disabled, mux_enable=host.mux_enable, fragment_setting=host.fragment_setting, + random_user_agent=host.random_user_agent, ) for host in modified_hosts ] db.commit() diff --git a/app/db/migrations/versions/31f92220c0d0_add_support_random_user_agent.py b/app/db/migrations/versions/31f92220c0d0_add_support_random_user_agent.py new file mode 100644 index 000000000..65c9b3412 --- /dev/null +++ b/app/db/migrations/versions/31f92220c0d0_add_support_random_user_agent.py @@ -0,0 +1,28 @@ +"""Add Support Random User-Agent + +Revision ID: 31f92220c0d0 +Revises: 4f045f53bef8 +Create Date: 2024-06-01 21:28:33.310627 + +""" +from alembic import op +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision = '31f92220c0d0' +down_revision = '4f045f53bef8' +branch_labels = None +depends_on = None + + +def upgrade() -> None: + # ### commands auto generated by Alembic - please adjust! ### + op.add_column('hosts', sa.Column('random_user_agent', sa.Boolean(), server_default='0', nullable=False)) + # ### end Alembic commands ### + + +def downgrade() -> None: + # ### commands auto generated by Alembic - please adjust! ### + op.drop_column('hosts', 'random_user_agent') + # ### end Alembic commands ### diff --git a/app/db/models.py b/app/db/models.py index 65d94bf9d..d19d90f6f 100644 --- a/app/db/models.py +++ b/app/db/models.py @@ -194,6 +194,7 @@ class ProxyHost(Base): is_disabled = Column(Boolean, nullable=True, default=False) mux_enable = Column(Boolean, nullable=False, default=False, server_default='0') fragment_setting = Column(String(100), nullable=True) + random_user_agent = Column(Boolean, nullable=False, default=False, server_default='0') class System(Base): diff --git a/app/models/proxy.py b/app/models/proxy.py index bef62f36e..c83ff9c21 100644 --- a/app/models/proxy.py +++ b/app/models/proxy.py @@ -149,6 +149,7 @@ class ProxyHost(BaseModel): is_disabled: Union[bool, None] = None mux_enable: Union[bool, None] = None fragment_setting: Optional[str] = Field(None, nullable=True) + random_user_agent: Union[bool, None] = None class Config: orm_mode = True diff --git a/app/subscription/share.py b/app/subscription/share.py index cd6e3869e..4d12de41e 100644 --- a/app/subscription/share.py +++ b/app/subscription/share.py @@ -282,7 +282,8 @@ def process_inbounds_and_tags( "ais": host["allowinsecure"] or inbound.get("allowinsecure", ""), "mux_enable": host["mux_enable"], - "fragment_setting": host["fragment_setting"] + "fragment_setting": host["fragment_setting"], + "random_user_agent": host["random_user_agent"], } ) diff --git a/app/subscription/singbox.py b/app/subscription/singbox.py index 7310baf85..5b5ffe8c0 100644 --- a/app/subscription/singbox.py +++ b/app/subscription/singbox.py @@ -1,8 +1,9 @@ import json +from random import choice from app.templates import render_template from app.subscription.funcs import get_grpc_gun -from config import SINGBOX_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE +from config import SINGBOX_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE, USER_AGENT_TEMPLATE class SingBoxConfiguration(str): @@ -11,6 +12,13 @@ def __init__(self): template = render_template(SINGBOX_SUBSCRIPTION_TEMPLATE) self.config = json.loads(template) self.mux_template = render_template(MUX_TEMPLATE) + temp_user_agent_data = render_template(USER_AGENT_TEMPLATE) + user_agent_data = json.loads(temp_user_agent_data) + + if 'list' in user_agent_data and isinstance(user_agent_data['list'], list): + self.user_agent_list = user_agent_data['list'] + else: + self.user_agent_list = [] def add_outbound(self, outbound_data): self.config["outbounds"].append(outbound_data) @@ -65,8 +73,8 @@ def tls_config(sni=None, fp=None, tls=None, pbk=None, return config - @staticmethod - def transport_config(transport_type='', + def transport_config(self, + transport_type='', host='', path='', method='', @@ -74,7 +82,8 @@ def transport_config(transport_type='', ping_timeout="15s", max_early_data=None, early_data_header_name=None, - permit_without_stream=False): + permit_without_stream=False, + random_user_agent: bool = False): transport_config = {} @@ -87,8 +96,12 @@ def transport_config(transport_type='', transport_config['path'] = path if method: transport_config['method'] = method + if host or random_user_agent: + transport_config['headers'] = {} if host: transport_config["host"] = [host] + if random_user_agent: + transport_config['headers']['User-Agent'] = choice(self.user_agent_list) if idle_timeout: transport_config['idle_timeout'] = idle_timeout if ping_timeout: @@ -97,8 +110,12 @@ def transport_config(transport_type='', elif transport_type == "ws": if path: transport_config['path'] = path + if host or random_user_agent: + transport_config['headers'] = {} if host: transport_config['headers'] = {'Host': host} + if random_user_agent: + transport_config['headers']['User-Agent'] = choice(self.user_agent_list) if max_early_data is not None: transport_config['max_early_data'] = max_early_data if early_data_header_name: @@ -118,6 +135,9 @@ def transport_config(transport_type='', transport_config['host'] = host if path: transport_config['path'] = path + if random_user_agent: + transport_config['headers'] = {} + transport_config['headers']['User-Agent'] = choice(self.user_agent_list) return transport_config @@ -139,6 +159,7 @@ def make_outbound(self, headers='', ais='', mux_enable: bool = False, + random_user_agent: bool = False, ): config = { @@ -173,7 +194,8 @@ def make_outbound(self, host=host, path=path, max_early_data=max_early_data, - early_data_header_name=early_data_header_name + early_data_header_name=early_data_header_name, + random_user_agent=random_user_agent, ) else: config["network"] = net @@ -215,7 +237,8 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): sid=inbound.get('sid', ''), headers=inbound['header_type'], ais=inbound.get('ais', ''), - mux_enable=inbound.get('mux_enable', False)) + mux_enable=inbound.get('mux_enable', False), + random_user_agent=inbound.get('random_user_agent', False),) if inbound['protocol'] == 'vmess': outbound['uuid'] = settings['id'] diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index ca9d900e1..625432db5 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -1,5 +1,6 @@ import base64 import json +from random import choice import urllib.parse as urlparse from typing import Union from uuid import UUID @@ -8,7 +9,7 @@ from app.subscription.funcs import get_grpc_gun, get_grpc_multi from app.templates import render_template -from config import (MUX_TEMPLATE, V2RAY_SUBSCRIPTION_TEMPLATE) +from config import (MUX_TEMPLATE, V2RAY_SUBSCRIPTION_TEMPLATE, USER_AGENT_TEMPLATE) class V2rayShareLink(str): @@ -341,6 +342,13 @@ def __init__(self): self.config = [] self.template = render_template(V2RAY_SUBSCRIPTION_TEMPLATE) self.mux_template = render_template(MUX_TEMPLATE) + temp_user_agent_data = render_template(USER_AGENT_TEMPLATE) + user_agent_data = json.loads(temp_user_agent_data) + + if 'list' in user_agent_data and isinstance(user_agent_data['list'], list): + self.user_agent_list = user_agent_data['list'] + else: + self.user_agent_list = [] def add_config(self, remarks, outbounds): json_template = json.loads(self.template) @@ -390,8 +398,8 @@ def reality_config(sni=None, fp=None, pbk=None, sid=None, spx=None): return realitySettings - @staticmethod - def ws_config(path=None, host=None): + + def ws_config(self, path=None, host=None, random_user_agent=None): wsSettings = {} wsSettings["headers"] = {} @@ -399,17 +407,21 @@ def ws_config(path=None, host=None): wsSettings["path"] = path if host: wsSettings["headers"]["Host"] = host + if random_user_agent: + wsSettings["headers"]["User-Agent"] = choice(self.user_agent_list) return wsSettings - @staticmethod - def httpupgrade_config(path=None, host=None): + def httpupgrade_config(self, path=None, host=None, random_user_agent=None): httpupgradeSettings = {} + httpupgradeSettings["headers"] = {} if path: httpupgradeSettings["path"] = path if host: httpupgradeSettings["host"] = host + if random_user_agent: + httpupgradeSettings["headers"]["User-Agent"] = choice(self.user_agent_list) return httpupgradeSettings @@ -427,8 +439,7 @@ def grpc_config(path=None, multiMode=False): return grpcSettings - @staticmethod - def tcp_http_config(path=None, host=None): + def tcp_http_config(self, path=None, host=None, random_user_agent=None): tcpSettings = {} if any((path, host)): @@ -440,7 +451,6 @@ def tcp_http_config(path=None, host=None): tcpSettings["header"]["request"]["headers"] = {} tcpSettings["header"]["request"]["method"] = "GET" - tcpSettings["header"]["request"]["headers"]["User-Agent"] = [] tcpSettings["header"]["request"]["headers"]["Accept-Encoding"] = ["gzip, deflate"] tcpSettings["header"]["request"]["headers"]["Connection"] = ["keep-alive"] tcpSettings["header"]["request"]["headers"]["Pragma"] = "no-cache" @@ -451,12 +461,17 @@ def tcp_http_config(path=None, host=None): if host: tcpSettings["header"]["request"]["headers"]["Host"] = [host] + if random_user_agent: + tcpSettings["header"]["request"]["headers"]["User-Agent"] = [choice(self.user_agent_list)] + else: + tcpSettings["header"]["request"]["headers"]["User-Agent"] = [] + return tcpSettings - @staticmethod - def h2_config(path=None, host=None): + def h2_config(self, path=None, host=None, random_user_agent=None): httpSettings = {} + httpSettings["headers"] = {} if path: httpSettings["path"] = path else: @@ -464,7 +479,9 @@ def h2_config(path=None, host=None): if host: httpSettings["host"] = [host] else: - httpSettings["host"] = {} + httpSettings["host"] = [] + if random_user_agent: + httpSettings["headers"]["User-Agent"] = [choice(self.user_agent_list)] return httpSettings @@ -647,24 +664,25 @@ def make_stream_setting(self, ais='', dialer_proxy='', multiMode: bool = False, + random_user_agent: bool = False, ): if net == "ws": - network_setting = self.ws_config(path=path, host=host) + network_setting = self.ws_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "grpc": network_setting = self.grpc_config(path=path, multiMode=multiMode) elif net == "h2": - network_setting = self.h2_config(path=path, host=host) + network_setting = self.h2_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "kcp": network_setting = self.kcp_config( path=path, host=host, header=headers) elif net == "tcp": - network_setting = self.tcp_http_config(path=path, host=host) + network_setting = self.tcp_http_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "quic": network_setting = self.quic_config( path=path, host=host, header=headers) elif net == "httpupgrade": - network_setting = self.httpupgrade_config(path=path, host=host) + network_setting = self.httpupgrade_config(path=path, host=host, random_user_agent=random_user_agent) if tls == "tls": tls_settings = self.tls_config(sni=sni, fp=fp, alpn=alpn, ais=ais) @@ -769,6 +787,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): ais=inbound.get('ais', ''), dialer_proxy=dialer_proxy, multiMode=multi_mode, + random_user_agent=inbound.get('random_user_agent', False), ) mux_json = json.loads(self.mux_template) diff --git a/app/templates/user_agent/default.json b/app/templates/user_agent/default.json new file mode 100644 index 000000000..abb85644c --- /dev/null +++ b/app/templates/user_agent/default.json @@ -0,0 +1,104 @@ +{ + "list":[ + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/125.0.0.0 Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/125.0.0.0 Safari/537.36 Edg/125.0.0.0", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/125.0.0.0 Safari/537.36", + "Mozilla/5.0 (iPhone; CPU iPhone OS 17_4_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.4.1 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/106.0.0.0 Safari/537.36 PageSpeedPlus/1.0.0", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 Safari/537.36 Edg/124.0.0.0", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/125.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:126.0) Gecko/20100101 Firefox/126.0", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36", + "Mozilla/5.0 (iPhone; CPU iPhone OS 17_4_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.4.1 Safari/605.1.15", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 11.6; rv:92.0) Gecko/20100101 Firefox/92.0", + "Mozilla/5.0 (iPhone; CPU iPhone OS 16_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.6 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/125.0.0.0 Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) obsidian/1.4.14 Chrome/114.0.5735.289 Electron/25.8.1 Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36", + "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:126.0) Gecko/20100101 Firefox/126.0", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.5 Safari/605.1.15", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/116.0.0.0 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36", + "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/125.0.0.0 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) SamsungBrowser/25.0 Chrome/121.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:125.0) Gecko/20100101 Firefox/125.0", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:125.0) Gecko/20100101 Firefox/125.0", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) obsidian/1.4.16 Chrome/114.0.5735.289 Electron/25.8.1 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36", + "Mozilla/5.0 (iPhone; CPU iPhone OS 17_2_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.2 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36 Edg/123.0.0.0", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.6 Safari/605.1.15", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) obsidian/1.5.3 Chrome/114.0.5735.289 Electron/25.8.1 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.1 Safari/605.1.15", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36", + "Mozilla/5.0 (iPhone; CPU iPhone OS 17_0_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.0 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36 Edg/122.0.0.0", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.2.1 Safari/605.1.15", + "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) obsidian/1.5.12 Chrome/120.0.6099.283 Electron/28.2.3 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) obsidian/1.4.16 Chrome/114.0.5735.289 Electron/25.8.1 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.0.0 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36 OPR/109.0.0.0", + "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/81.0.4044.129 Safari/537.36", + "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/116.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.1 Safari/605.1.15", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.3.1 Safari/605.1.15", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) obsidian/1.4.13 Chrome/114.0.5735.289 Electron/25.8.1 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.0 Safari/605.1.15", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.2 Safari/605.1.15", + "Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:109.0) Gecko/20100101 Firefox/115.0", + "Mozilla/5.0 (iPhone; CPU iPhone OS 17_5_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.5 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (iPhone; CPU iPhone OS 17_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.3 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (iPhone; CPU iPhone OS 16_1_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.1 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (iPhone; CPU iPhone OS 17_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.3.1 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:124.0) Gecko/20100101 Firefox/124.0", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.3 Safari/605.1.15", + "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/126.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) obsidian/1.4.13 Chrome/114.0.5735.289 Electron/25.8.1 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.4 Safari/605.1.15", + "Mozilla/5.0 (iPhone; CPU iPhone OS 16_6 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.6 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Safari/605.1.15", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36", + "Mozilla/5.0 (iPhone; CPU iPhone OS 17_1_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.1.1 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36 Edg/120.0.0.0", + "Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.5 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) obsidian/1.5.12 Chrome/120.0.6099.283 Electron/28.2.3 Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/117.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.1.2 Safari/605.1.15", + "Mozilla/5.0 (iPhone; CPU iPhone OS 16_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.2 Mobile/15E148 Safari/604.1", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.5 Safari/605.1.15", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36 Edg/121.0.0.0", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:124.0) Gecko/20100101 Firefox/124.0", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) obsidian/1.5.3 Chrome/114.0.5735.289 Electron/25.8.1 Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.67 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.3 Safari/605.1.15", + "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36", + "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36", + "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 Safari/537.36 Edg/124.0.0.0", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/117.0.0.0 Mobile Safari/537.36", + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36", + "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) SamsungBrowser/24.0 Chrome/117.0.0.0 Mobile Safari/537.36" + ] +} \ No newline at end of file diff --git a/app/xray/__init__.py b/app/xray/__init__.py index 8f13d9424..1bc23fe60 100644 --- a/app/xray/__init__.py +++ b/app/xray/__init__.py @@ -61,7 +61,8 @@ def hosts(storage: dict): else host.security.value, "allowinsecure": host.allowinsecure, "mux_enable": host.mux_enable, - "fragment_setting": host.fragment_setting + "fragment_setting": host.fragment_setting, + "random_user_agent": host.random_user_agent, } for host in inbound_hosts if not host.is_disabled ] diff --git a/config.py b/config.py index fa22f994b..1565bc903 100755 --- a/config.py +++ b/config.py @@ -50,6 +50,8 @@ SINGBOX_SUBSCRIPTION_TEMPLATE = config("SINGBOX_SUBSCRIPTION_TEMPLATE", default="singbox/default.json") MUX_TEMPLATE = config("MUX_TEMPLATE", default="mux/default.json") V2RAY_SUBSCRIPTION_TEMPLATE = config("V2RAY_SUBSCRIPTION_TEMPLATE", default="v2ray/default.json") +USER_AGENT_TEMPLATE = config("USER_AGENT_TEMPLATE", default="user_agent/default.json") + USE_CUSTOM_JSON_DEFAULT = config("USE_CUSTOM_JSON_DEFAULT", default=False, cast=bool) USE_CUSTOM_JSON_FOR_V2RAYN = config("USE_CUSTOM_JSON_FOR_V2RAYN", default=False, cast=bool) From 0f524526a2fda3dfe8c7e27a7f0427d68ae57a8e Mon Sep 17 00:00:00 2001 From: SaintShit Date: Tue, 2 Jul 2024 01:26:08 +0330 Subject: [PATCH 25/52] feat: random user agent for clash configuration --- app/subscription/clash.py | 44 ++++++++++++++++++++++++++++----------- 1 file changed, 32 insertions(+), 12 deletions(-) diff --git a/app/subscription/clash.py b/app/subscription/clash.py index 982bb9d8a..ae25d2b6c 100644 --- a/app/subscription/clash.py +++ b/app/subscription/clash.py @@ -1,9 +1,12 @@ -import yaml import json -from app.templates import render_template -from app.subscription.funcs import get_grpc_gun +from random import choice -from config import CLASH_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE +import yaml + +from app.subscription.funcs import get_grpc_gun +from app.templates import render_template +from config import (CLASH_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE, + USER_AGENT_TEMPLATE) class ClashConfiguration(object): @@ -16,6 +19,13 @@ def __init__(self): } self.proxy_remarks = [] self.mux_template = render_template(MUX_TEMPLATE) + temp_user_agent_data = render_template(USER_AGENT_TEMPLATE) + user_agent_data = json.loads(temp_user_agent_data) + + if 'list' in user_agent_data and isinstance(user_agent_data['list'], list): + self.user_agent_list = user_agent_data['list'] + else: + self.user_agent_list = [] def render(self): return yaml.dump( @@ -60,8 +70,8 @@ def make_node(self, udp: bool = True, alpn: str = '', ais: bool = '', - mux_enable: bool = False): - + mux_enable: bool = False, + random_user_agent: bool = False): if network in ["grpc", "gun"]: path = get_grpc_gun(path) @@ -113,12 +123,18 @@ def make_node(self, if host: net_opts['method'] = 'GET' net_opts['Host'] = host + if random_user_agent: + net_opts['header'] = {"User-Agent": choice(self.user_agent_list)} if network == 'ws' or network == 'httpupgrade': if path: net_opts['path'] = path - if host: - net_opts['headers'] = {"Host": host} + if host or random_user_agent: + net_opts['headers'] = {} + if host: + net_opts['headers']["Host"] = host + if random_user_agent: + net_opts['headers']["User-Agent"] = choice(self.user_agent_list) if max_early_data: net_opts['max-early-data'] = max_early_data net_opts['early-data-header-name'] = early_data_header_name @@ -169,7 +185,8 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): udp=True, alpn=inbound.get('alpn', ''), ais=inbound.get('ais', ''), - mux_enable=inbound.get('mux_enable', '') + mux_enable=inbound.get('mux_enable', ''), + random_user_agent=inbound.get("random_user_agent") ) if inbound['protocol'] == 'vmess': @@ -209,7 +226,8 @@ def make_node(self, pbk: str = '', sid: str = '', ais: bool = '', - mux_enable: bool = False): + mux_enable: bool = False, + random_user_agent: bool = False): node = super().make_node( name=name, type=type, @@ -224,7 +242,8 @@ def make_node(self, udp=udp, alpn=alpn, ais=ais, - mux_enable=mux_enable + mux_enable=mux_enable, + random_user_agent=random_user_agent ) if fp: node['client-fingerprint'] = fp @@ -251,7 +270,8 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): pbk=inbound.get('pbk', ''), sid=inbound.get('sid', ''), ais=inbound.get('ais', ''), - mux_enable=inbound.get('mux_enable', '') + mux_enable=inbound.get('mux_enable', ''), + random_user_agent=inbound.get("random_user_agent") ) if inbound['protocol'] == 'vmess': From 4609962e16271dea7388ef9f7c16b11222227a6e Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Tue, 2 Jul 2024 22:13:29 +0330 Subject: [PATCH 26/52] fix: sing-box mux enable --- app/subscription/singbox.py | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/app/subscription/singbox.py b/app/subscription/singbox.py index 5b5ffe8c0..d237287c6 100644 --- a/app/subscription/singbox.py +++ b/app/subscription/singbox.py @@ -204,10 +204,12 @@ def make_outbound(self, config['tls'] = self.tls_config(sni=sni, fp=fp, tls=tls, pbk=pbk, sid=sid, alpn=alpn, ais=ais) - if mux_enable: - mux_json = json.loads(self.mux_template) - mux_config = mux_json["sing-box"] - config['multiplex'] = mux_config + + mux_json = json.loads(self.mux_template) + mux_config = mux_json["sing-box"] + + config['multiplex'] = mux_config + if config['multiplex']["enabled"]: config['multiplex']["enabled"] = mux_enable return config From 8485fda0faadb50cd3a24d87c164d7e3db6e90a9 Mon Sep 17 00:00:00 2001 From: SaintShit Date: Wed, 3 Jul 2024 10:44:56 +0330 Subject: [PATCH 27/52] fix: improve the speed of xray.config.include_db_users --- app/xray/config.py | 109 ++++++++++++++++++++++++++++----------------- 1 file changed, 68 insertions(+), 41 deletions(-) diff --git a/app/xray/config.py b/app/xray/config.py index 6cb82b616..46c3620bb 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -1,14 +1,17 @@ from __future__ import annotations import json +from collections import defaultdict from copy import deepcopy from pathlib import PosixPath from typing import Union import commentjson +from sqlalchemy import func -from app.db import GetDB, crud -from app.models.proxy import ProxySettings, ProxyTypes +from app.db import GetDB +from app.db import models as db_models +from app.models.proxy import ProxyTypes from app.models.user import UserStatus from app.utils.crypto import get_cert_SANs from config import DEBUG, XRAY_EXCLUDE_INBOUND_TAGS, XRAY_FALLBACKS_INBOUND_TAG @@ -45,7 +48,6 @@ def __init__(self, self.inbounds_by_protocol = {} self.inbounds_by_tag = {} self._fallbacks_inbound = self.get_inbound(XRAY_FALLBACKS_INBOUND_TAG) - self._addr_clients_by_tag = {} self._resolve_inbounds() self._apply_api() @@ -119,6 +121,8 @@ def _validate(self): for inbound in self['inbounds']: if not inbound.get("tag"): raise ValueError("all inbounds must have a unique tag") + if ',' in inbound.get("tag"): + raise ValueError("character «,» is not allowed in inbound tag") for outbound in self['outbounds']: if not outbound.get("tag"): raise ValueError("all outbounds must have a unique tag") @@ -135,8 +139,6 @@ def _resolve_inbounds(self): inbound['settings'] = {} if not inbound['settings'].get('clients'): inbound['settings']['clients'] = [] - self._addr_clients_by_tag[inbound['tag'] - ] = inbound['settings']['clients'] settings = { "tag": inbound["tag"], @@ -298,30 +300,6 @@ def _resolve_inbounds(self): except KeyError: self.inbounds_by_protocol[inbound['protocol']] = [settings] - def add_inbound_client(self, inbound_tag: str, email: str, settings: dict): - inbound = self.inbounds_by_tag.get(inbound_tag, {}) - client = {"email": email, **settings} - - # XTLS currently only supports transmission methods of TCP and mKCP - if client.get('flow') and ( - inbound.get('network', 'tcp') not in ('tcp', 'kcp') - or - ( - inbound.get('network', 'tcp') in ('tcp', 'kcp') - and - inbound.get('tls') not in ('tls', 'reality') - ) - or - inbound.get('header_type') == 'http' - ): - del client['flow'] - - try: - self._addr_clients_by_tag[inbound_tag].append(client) - except KeyError: - return - return client - def get_inbound(self, tag) -> dict: for inbound in self['inbounds']: if inbound['tag'] == tag: @@ -342,18 +320,67 @@ def include_db_users(self) -> XRayConfig: config = self.copy() with GetDB() as db: - for user in crud.get_users(db, status=[UserStatus.active, - UserStatus.on_hold]): - proxies_settings = { - p.type: ProxySettings.from_dict( - p.type, p.settings).dict(no_obj=True) - for p in user.proxies - } - for proxy_type, inbound_tags in user.inbounds.items(): - for inbound_tag in inbound_tags: - config.add_inbound_client(inbound_tag, - f"{user.id}.{user.username}", - proxies_settings[proxy_type]) + query = db.query( + db_models.User.id, + db_models.User.username, + func.lower(db_models.Proxy.type).label('type'), + db_models.Proxy.settings, + func.group_concat(db_models.excluded_inbounds_association.c.inbound_tag).label('excluded_inbound_tags') + ).join( + db_models.Proxy, db_models.User.id == db_models.Proxy.user_id + ).outerjoin( + db_models.excluded_inbounds_association, db_models.Proxy.id == db_models.excluded_inbounds_association.c.proxy_id + ).filter( + db_models.User.status.in_([UserStatus.active, UserStatus.on_hold]) + ).group_by( + func.lower(db_models.Proxy.type), + db_models.User.id + ) + result = query.all() + + grouped_data = defaultdict(list) + + for row in result: + grouped_data[row["type"]].append(( + row["id"], + row["username"], + row["settings"], + [i for i in row['excluded_inbound_tags'].split(',') if i] if row['excluded_inbound_tags'] else None + )) + + for proxy_type, rows in grouped_data.items(): + + inbounds = self.inbounds_by_protocol[proxy_type] + + for inbound in inbounds: + clients = config.get_inbound(inbound['tag'])['settings']['clients'] + + for row in rows: + user_id, username, settings, excluded_inbound_tags = row + + if excluded_inbound_tags and inbound['tag'] in excluded_inbound_tags: + continue + + client = { + "email": f"{user_id}.{username}", + **settings + } + + # XTLS currently only supports transmission methods of TCP and mKCP + if client.get('flow') and ( + inbound.get('network', 'tcp') not in ('tcp', 'kcp') + or + ( + inbound.get('network', 'tcp') in ('tcp', 'kcp') + and + inbound.get('tls') not in ('tls', 'reality') + ) + or + inbound.get('header_type') == 'http' + ): + del client['flow'] + + clients.append(client) if DEBUG: with open('generated_config-debug.json', 'w') as f: From e228068a492b67ed28ad86350242ce6bfd5bb6a7 Mon Sep 17 00:00:00 2001 From: SaintShit Date: Wed, 3 Jul 2024 10:58:22 +0330 Subject: [PATCH 28/52] fix: skip proxies that has no active inbound --- app/xray/config.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/app/xray/config.py b/app/xray/config.py index 46c3620bb..0d904c4c3 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -350,7 +350,9 @@ def include_db_users(self) -> XRayConfig: for proxy_type, rows in grouped_data.items(): - inbounds = self.inbounds_by_protocol[proxy_type] + inbounds = self.inbounds_by_protocol.get('proxy_type') + if not inbounds: + continue for inbound in inbounds: clients = config.get_inbound(inbound['tag'])['settings']['clients'] From 987c16def64bc22075f3c0e0394909996f5054ec Mon Sep 17 00:00:00 2001 From: SaintShit Date: Wed, 3 Jul 2024 11:23:54 +0330 Subject: [PATCH 29/52] fix: stupid bug --- app/xray/config.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/xray/config.py b/app/xray/config.py index 0d904c4c3..e77b4682f 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -350,7 +350,7 @@ def include_db_users(self) -> XRayConfig: for proxy_type, rows in grouped_data.items(): - inbounds = self.inbounds_by_protocol.get('proxy_type') + inbounds = self.inbounds_by_protocol.get(proxy_type) if not inbounds: continue From cdff3efd53210664ee1d395c7ce64aaeb8818d31 Mon Sep 17 00:00:00 2001 From: Erfan <143827987+erfjab@users.noreply.github.com> Date: Wed, 3 Jul 2024 12:41:10 +0330 Subject: [PATCH 30/52] add random sub configs status to .env.example --- .env.example | 1 + 1 file changed, 1 insertion(+) diff --git a/.env.example b/.env.example index dfb3faf23..f5c48ed8a 100644 --- a/.env.example +++ b/.env.example @@ -49,6 +49,7 @@ UVICORN_PORT = 8000 # SUB_PROFILE_TITLE = "Susbcription" # SUB_SUPPORT_URL = "https://t.me/support" # SUB_UPDATE_INTERVAL = "12" +# RANDOMIZE_SUB_CONFIGS_STATUS = True # SQLALCHEMY_DATABASE_URL = "sqlite:///db.sqlite3" From 1bdc60ee5142572f3b1c2fa13c6f31d06bc5306b Mon Sep 17 00:00:00 2001 From: Erfan <143827987+erfjab@users.noreply.github.com> Date: Wed, 3 Jul 2024 12:42:19 +0330 Subject: [PATCH 31/52] add random sub configs status to config.py --- config.py | 1 + 1 file changed, 1 insertion(+) diff --git a/config.py b/config.py index 1565bc903..6c1ee54bc 100755 --- a/config.py +++ b/config.py @@ -97,6 +97,7 @@ SUB_UPDATE_INTERVAL = config("SUB_UPDATE_INTERVAL", default="12") SUB_SUPPORT_URL = config("SUB_SUPPORT_URL", default="https://t.me/") SUB_PROFILE_TITLE = config("SUB_PROFILE_TITLE", default="Subscription") +RANDOM_SUB_CONFIGS_STATUS = config("RANDOM_SUB_CONFIGS_STATUS", default=False, cast=bool) # discord webhook log DISCORD_WEBHOOK_URL = config("DISCORD_WEBHOOK_URL", default="") From 89837c1f1d4d45965aabee38b696c39ecba5af8e Mon Sep 17 00:00:00 2001 From: Erfan <143827987+erfjab@users.noreply.github.com> Date: Wed, 3 Jul 2024 12:44:14 +0330 Subject: [PATCH 32/52] add randomize sub configs to share.py --- app/subscription/share.py | 40 ++++++++++++++++++++++++++++++++++++++- 1 file changed, 39 insertions(+), 1 deletion(-) diff --git a/app/subscription/share.py b/app/subscription/share.py index 4d12de41e..c9eb9a520 100644 --- a/app/subscription/share.py +++ b/app/subscription/share.py @@ -1,6 +1,8 @@ import base64 import random import secrets +import yaml +import json from datetime import datetime as dt from datetime import timedelta from typing import TYPE_CHECKING, List, Literal, Union @@ -17,7 +19,7 @@ from config import (ACTIVE_STATUS_TEXT, DISABLED_STATUS_TEXT, EXPIRED_STATUS_TEXT, LIMITED_STATUS_TEXT, - ONHOLD_STATUS_TEXT) + ONHOLD_STATUS_TEXT, RANDOM_SUB_CONFIGS_STATUS) SERVER_IP = get_public_ip() SERVER_IPV6 = get_public_ipv6() @@ -95,7 +97,40 @@ def generate_v2ray_json_subscription( inbounds, proxies, format_variables, conf=conf ) +def randomize_sub_config( + config: str, config_format: str +) -> str: + + if config_format == "v2ray": + config = config.split("\n") + random.shuffle(config) + config = "\n".join(config) + + elif config_format in ("clash-meta", "clash"): + config = yaml.safe_load(config) + random.shuffle(config['proxies']) + for group in config['proxy-groups']: + if group['name'] == '♻️ Automatic': + group['proxies'] = [proxy['name'] for proxy in config['proxies']] + config = yaml.dump(config, allow_unicode=True, sort_keys=False) + + elif config_format == "sing-box": + config = json.loads(config) + outbounds = config['outbounds'] + main_outbounds = [ob for ob in outbounds if ob['type'] in {'selector', 'urltest'}] + other_outbounds = [ob for ob in outbounds if ob['type'] not in {'selector', 'urltest', 'direct', 'block', 'dns'}] + random.shuffle(other_outbounds) + proxy_names = [ob['tag'] for ob in other_outbounds] + for ob in main_outbounds: + ob['outbounds'] = ['Best Latency'] + proxy_names if ob['type'] == 'selector' else proxy_names + config['outbounds'] = main_outbounds + other_outbounds + [ob for ob in outbounds if ob['type'] in {'direct', 'block', 'dns'}] + config = json.dumps(config, indent=4) + elif config_format == "v2ray-json": + random.shuffle(config) + + return config + def generate_subscription( user: "UserResponse", config_format: Literal["v2ray", "clash-meta", "clash", "sing-box", "outline", "v2ray-json"], @@ -122,6 +157,9 @@ def generate_subscription( else: raise ValueError(f'Unsupported format "{config_format}"') + if RANDOM_SUB_CONFIGS_STATUS is not False: + config = randomize_sub_config(config, config_format) + if as_base64: config = base64.b64encode(config.encode()).decode() From 6a434d595d50874ff3a984af4e69f2ebbc5a3cd6 Mon Sep 17 00:00:00 2001 From: Mohammad20000 Date: Wed, 3 Jul 2024 13:36:28 +0330 Subject: [PATCH 33/52] Fix: subscription display in Hiddify --- app/views/subscription.py | 11 +++++------ 1 file changed, 5 insertions(+), 6 deletions(-) diff --git a/app/views/subscription.py b/app/views/subscription.py index d587d7060..e304311a4 100644 --- a/app/views/subscription.py +++ b/app/views/subscription.py @@ -8,19 +8,19 @@ from app import app from app.db import Session, crud, get_db from app.models.user import UserResponse +from app.subscription.share import encode_title, generate_subscription from app.templates import render_template from app.utils.jwt import get_subscription_payload -from app.subscription.share import encode_title, generate_subscription from config import ( SUB_PROFILE_TITLE, SUB_SUPPORT_URL, SUB_UPDATE_INTERVAL, SUBSCRIPTION_PAGE_TEMPLATE, - XRAY_SUBSCRIPTION_PATH, USE_CUSTOM_JSON_DEFAULT, + USE_CUSTOM_JSON_FOR_STREISAND, USE_CUSTOM_JSON_FOR_V2RAYN, USE_CUSTOM_JSON_FOR_V2RAYNG, - USE_CUSTOM_JSON_FOR_STREISAND + XRAY_SUBSCRIPTION_PATH ) @@ -192,8 +192,8 @@ def get_subscription_user_info(user: UserResponse) -> dict: return { "upload": 0, "download": user.used_traffic, - "total": user.data_limit, - "expire": user.expire, + "total": user.data_limit if user.data_limit is not None else 0, + "expire": user.expire if user.expire is not None else 0, } sub = get_subscription_payload(token) @@ -218,7 +218,6 @@ def get_subscription_user_info(user: UserResponse) -> dict: "subscription-userinfo": "; ".join( f"{key}={val}" for key, val in get_subscription_user_info(user).items() - if val is not None ) } From a964458b4293a6f2fb5da1c59ec9b82000883e24 Mon Sep 17 00:00:00 2001 From: SaintShit Date: Wed, 3 Jul 2024 13:54:46 +0330 Subject: [PATCH 34/52] fix: update users sub_last_user_agent on /sub/client_type endpoint --- app/views/subscription.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/app/views/subscription.py b/app/views/subscription.py index e304311a4..0776ca19c 100644 --- a/app/views/subscription.py +++ b/app/views/subscription.py @@ -183,6 +183,7 @@ def user_subscription_with_client_type( request: Request, client_type: str = Path(..., regex="sing-box|clash-meta|clash|outline|v2ray|v2ray-json"), db: Session = Depends(get_db), + user_agent: str = Header(default="") ): """ Subscription link, v2ray, clash, sing-box, outline and clash-meta supported @@ -221,6 +222,8 @@ def get_subscription_user_info(user: UserResponse) -> dict: ) } + crud.update_user_sub(db, dbuser, user_agent) + if client_type == "clash-meta": conf = generate_subscription(user=user, config_format="clash-meta", as_base64=False) return Response(content=conf, media_type="text/yaml", headers=response_headers) From 5f9b068e5636d1e4dbf90ebb681367d5890b701e Mon Sep 17 00:00:00 2001 From: SaintShit Date: Wed, 3 Jul 2024 14:16:50 +0330 Subject: [PATCH 35/52] fix: update xray proto files to v1.8.16 --- xray_api/proto/app/commander/config_pb2.py | 8 ++-- .../proto/transport/internet/config_pb2.py | 9 ++-- .../internet/splithttp/config_pb2.py | 48 +++++++++++++++++++ .../internet/splithttp/config_pb2_grpc.py | 4 ++ 4 files changed, 61 insertions(+), 8 deletions(-) create mode 100644 xray_api/proto/transport/internet/splithttp/config_pb2.py create mode 100644 xray_api/proto/transport/internet/splithttp/config_pb2_grpc.py diff --git a/xray_api/proto/app/commander/config_pb2.py b/xray_api/proto/app/commander/config_pb2.py index bd9bffa75..8310a393e 100644 --- a/xray_api/proto/app/commander/config_pb2.py +++ b/xray_api/proto/app/commander/config_pb2.py @@ -15,7 +15,7 @@ from xray_api.proto.common.serial import typed_message_pb2 as common_dot_serial_dot_typed__message__pb2 -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1a\x61pp/commander/config.proto\x12\x12xray.app.commander\x1a!common/serial/typed_message.proto\"H\n\x06\x43onfig\x12\x0b\n\x03tag\x18\x01 \x01(\t\x12\x31\n\x07service\x18\x02 \x03(\x0b\x32 .xray.common.serial.TypedMessage\"\x12\n\x10ReflectionConfigBX\n\x16\x63om.xray.app.commanderP\x01Z\'github.com/xtls/xray-core/app/commander\xaa\x02\x12Xray.App.Commanderb\x06proto3') +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1a\x61pp/commander/config.proto\x12\x12xray.app.commander\x1a!common/serial/typed_message.proto\"X\n\x06\x43onfig\x12\x0b\n\x03tag\x18\x01 \x01(\t\x12\x0e\n\x06listen\x18\x03 \x01(\t\x12\x31\n\x07service\x18\x02 \x03(\x0b\x32 .xray.common.serial.TypedMessage\"\x12\n\x10ReflectionConfigBX\n\x16\x63om.xray.app.commanderP\x01Z\'github.com/xtls/xray-core/app/commander\xaa\x02\x12Xray.App.Commanderb\x06proto3') @@ -40,7 +40,7 @@ DESCRIPTOR._options = None DESCRIPTOR._serialized_options = b'\n\026com.xray.app.commanderP\001Z\'github.com/xtls/xray-core/app/commander\252\002\022Xray.App.Commander' _CONFIG._serialized_start=85 - _CONFIG._serialized_end=157 - _REFLECTIONCONFIG._serialized_start=159 - _REFLECTIONCONFIG._serialized_end=177 + _CONFIG._serialized_end=173 + _REFLECTIONCONFIG._serialized_start=175 + _REFLECTIONCONFIG._serialized_end=193 # @@protoc_insertion_point(module_scope) diff --git a/xray_api/proto/transport/internet/config_pb2.py b/xray_api/proto/transport/internet/config_pb2.py index 2983bdf1d..cb75d9dfb 100644 --- a/xray_api/proto/transport/internet/config_pb2.py +++ b/xray_api/proto/transport/internet/config_pb2.py @@ -16,7 +16,7 @@ from xray_api.proto.common.serial import typed_message_pb2 as common_dot_serial_dot_typed__message__pb2 -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1ftransport/internet/config.proto\x12\x17xray.transport.internet\x1a!common/serial/typed_message.proto\"\x9e\x01\n\x0fTransportConfig\x12@\n\x08protocol\x18\x01 \x01(\x0e\x32*.xray.transport.internet.TransportProtocolB\x02\x18\x01\x12\x15\n\rprotocol_name\x18\x03 \x01(\t\x12\x32\n\x08settings\x18\x02 \x01(\x0b\x32 .xray.common.serial.TypedMessage\"\xc1\x02\n\x0cStreamConfig\x12@\n\x08protocol\x18\x01 \x01(\x0e\x32*.xray.transport.internet.TransportProtocolB\x02\x18\x01\x12\x15\n\rprotocol_name\x18\x05 \x01(\t\x12\x44\n\x12transport_settings\x18\x02 \x03(\x0b\x32(.xray.transport.internet.TransportConfig\x12\x15\n\rsecurity_type\x18\x03 \x01(\t\x12;\n\x11security_settings\x18\x04 \x03(\x0b\x32 .xray.common.serial.TypedMessage\x12>\n\x0fsocket_settings\x18\x06 \x01(\x0b\x32%.xray.transport.internet.SocketConfig\"7\n\x0bProxyConfig\x12\x0b\n\x03tag\x18\x01 \x01(\t\x12\x1b\n\x13transportLayerProxy\x18\x02 \x01(\x08\"\xce\x04\n\x0cSocketConfig\x12\x0c\n\x04mark\x18\x01 \x01(\x05\x12\x0b\n\x03tfo\x18\x02 \x01(\x05\x12@\n\x06tproxy\x18\x03 \x01(\x0e\x32\x30.xray.transport.internet.SocketConfig.TProxyMode\x12%\n\x1dreceive_original_dest_address\x18\x04 \x01(\x08\x12\x14\n\x0c\x62ind_address\x18\x05 \x01(\x0c\x12\x11\n\tbind_port\x18\x06 \x01(\r\x12\x1d\n\x15\x61\x63\x63\x65pt_proxy_protocol\x18\x07 \x01(\x08\x12@\n\x0f\x64omain_strategy\x18\x08 \x01(\x0e\x32\'.xray.transport.internet.DomainStrategy\x12\x14\n\x0c\x64ialer_proxy\x18\t \x01(\t\x12\x1f\n\x17tcp_keep_alive_interval\x18\n \x01(\x05\x12\x1b\n\x13tcp_keep_alive_idle\x18\x0b \x01(\x05\x12\x16\n\x0etcp_congestion\x18\x0c \x01(\t\x12\x11\n\tinterface\x18\r \x01(\t\x12\x0e\n\x06v6only\x18\x0e \x01(\x08\x12\x18\n\x10tcp_window_clamp\x18\x0f \x01(\x05\x12\x18\n\x10tcp_user_timeout\x18\x10 \x01(\x05\x12\x13\n\x0btcp_max_seg\x18\x11 \x01(\x05\x12\x14\n\x0ctcp_no_delay\x18\x12 \x01(\x08\x12\x11\n\ttcp_mptcp\x18\x13 \x01(\x08\"/\n\nTProxyMode\x12\x07\n\x03Off\x10\x00\x12\n\n\x06TProxy\x10\x01\x12\x0c\n\x08Redirect\x10\x02*k\n\x11TransportProtocol\x12\x07\n\x03TCP\x10\x00\x12\x07\n\x03UDP\x10\x01\x12\x08\n\x04MKCP\x10\x02\x12\r\n\tWebSocket\x10\x03\x12\x08\n\x04HTTP\x10\x04\x12\x10\n\x0c\x44omainSocket\x10\x05\x12\x0f\n\x0bHTTPUpgrade\x10\x06*\xa9\x01\n\x0e\x44omainStrategy\x12\t\n\x05\x41S_IS\x10\x00\x12\n\n\x06USE_IP\x10\x01\x12\x0b\n\x07USE_IP4\x10\x02\x12\x0b\n\x07USE_IP6\x10\x03\x12\x0c\n\x08USE_IP46\x10\x04\x12\x0c\n\x08USE_IP64\x10\x05\x12\x0c\n\x08\x46ORCE_IP\x10\x06\x12\r\n\tFORCE_IP4\x10\x07\x12\r\n\tFORCE_IP6\x10\x08\x12\x0e\n\nFORCE_IP46\x10\t\x12\x0e\n\nFORCE_IP64\x10\nBg\n\x1b\x63om.xray.transport.internetP\x01Z,github.com/xtls/xray-core/transport/internet\xaa\x02\x17Xray.Transport.Internetb\x06proto3') +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1ftransport/internet/config.proto\x12\x17xray.transport.internet\x1a!common/serial/typed_message.proto\"\x9e\x01\n\x0fTransportConfig\x12@\n\x08protocol\x18\x01 \x01(\x0e\x32*.xray.transport.internet.TransportProtocolB\x02\x18\x01\x12\x15\n\rprotocol_name\x18\x03 \x01(\t\x12\x32\n\x08settings\x18\x02 \x01(\x0b\x32 .xray.common.serial.TypedMessage\"\xc1\x02\n\x0cStreamConfig\x12@\n\x08protocol\x18\x01 \x01(\x0e\x32*.xray.transport.internet.TransportProtocolB\x02\x18\x01\x12\x15\n\rprotocol_name\x18\x05 \x01(\t\x12\x44\n\x12transport_settings\x18\x02 \x03(\x0b\x32(.xray.transport.internet.TransportConfig\x12\x15\n\rsecurity_type\x18\x03 \x01(\t\x12;\n\x11security_settings\x18\x04 \x03(\x0b\x32 .xray.common.serial.TypedMessage\x12>\n\x0fsocket_settings\x18\x06 \x01(\x0b\x32%.xray.transport.internet.SocketConfig\"7\n\x0bProxyConfig\x12\x0b\n\x03tag\x18\x01 \x01(\t\x12\x1b\n\x13transportLayerProxy\x18\x02 \x01(\x08\"\xce\x04\n\x0cSocketConfig\x12\x0c\n\x04mark\x18\x01 \x01(\x05\x12\x0b\n\x03tfo\x18\x02 \x01(\x05\x12@\n\x06tproxy\x18\x03 \x01(\x0e\x32\x30.xray.transport.internet.SocketConfig.TProxyMode\x12%\n\x1dreceive_original_dest_address\x18\x04 \x01(\x08\x12\x14\n\x0c\x62ind_address\x18\x05 \x01(\x0c\x12\x11\n\tbind_port\x18\x06 \x01(\r\x12\x1d\n\x15\x61\x63\x63\x65pt_proxy_protocol\x18\x07 \x01(\x08\x12@\n\x0f\x64omain_strategy\x18\x08 \x01(\x0e\x32\'.xray.transport.internet.DomainStrategy\x12\x14\n\x0c\x64ialer_proxy\x18\t \x01(\t\x12\x1f\n\x17tcp_keep_alive_interval\x18\n \x01(\x05\x12\x1b\n\x13tcp_keep_alive_idle\x18\x0b \x01(\x05\x12\x16\n\x0etcp_congestion\x18\x0c \x01(\t\x12\x11\n\tinterface\x18\r \x01(\t\x12\x0e\n\x06v6only\x18\x0e \x01(\x08\x12\x18\n\x10tcp_window_clamp\x18\x0f \x01(\x05\x12\x18\n\x10tcp_user_timeout\x18\x10 \x01(\x05\x12\x13\n\x0btcp_max_seg\x18\x11 \x01(\x05\x12\x14\n\x0ctcp_no_delay\x18\x12 \x01(\x08\x12\x11\n\ttcp_mptcp\x18\x13 \x01(\x08\"/\n\nTProxyMode\x12\x07\n\x03Off\x10\x00\x12\n\n\x06TProxy\x10\x01\x12\x0c\n\x08Redirect\x10\x02*z\n\x11TransportProtocol\x12\x07\n\x03TCP\x10\x00\x12\x07\n\x03UDP\x10\x01\x12\x08\n\x04MKCP\x10\x02\x12\r\n\tWebSocket\x10\x03\x12\x08\n\x04HTTP\x10\x04\x12\x10\n\x0c\x44omainSocket\x10\x05\x12\x0f\n\x0bHTTPUpgrade\x10\x06\x12\r\n\tSplitHTTP\x10\x07*\xa9\x01\n\x0e\x44omainStrategy\x12\t\n\x05\x41S_IS\x10\x00\x12\n\n\x06USE_IP\x10\x01\x12\x0b\n\x07USE_IP4\x10\x02\x12\x0b\n\x07USE_IP6\x10\x03\x12\x0c\n\x08USE_IP46\x10\x04\x12\x0c\n\x08USE_IP64\x10\x05\x12\x0c\n\x08\x46ORCE_IP\x10\x06\x12\r\n\tFORCE_IP4\x10\x07\x12\r\n\tFORCE_IP6\x10\x08\x12\x0e\n\nFORCE_IP46\x10\t\x12\x0e\n\nFORCE_IP64\x10\nBg\n\x1b\x63om.xray.transport.internetP\x01Z,github.com/xtls/xray-core/transport/internet\xaa\x02\x17Xray.Transport.Internetb\x06proto3') _TRANSPORTPROTOCOL = DESCRIPTOR.enum_types_by_name['TransportProtocol'] TransportProtocol = enum_type_wrapper.EnumTypeWrapper(_TRANSPORTPROTOCOL) @@ -29,6 +29,7 @@ HTTP = 4 DomainSocket = 5 HTTPUpgrade = 6 +SplitHTTP = 7 AS_IS = 0 USE_IP = 1 USE_IP4 = 2 @@ -84,9 +85,9 @@ _STREAMCONFIG.fields_by_name['protocol']._options = None _STREAMCONFIG.fields_by_name['protocol']._serialized_options = b'\030\001' _TRANSPORTPROTOCOL._serialized_start=1230 - _TRANSPORTPROTOCOL._serialized_end=1337 - _DOMAINSTRATEGY._serialized_start=1340 - _DOMAINSTRATEGY._serialized_end=1509 + _TRANSPORTPROTOCOL._serialized_end=1352 + _DOMAINSTRATEGY._serialized_start=1355 + _DOMAINSTRATEGY._serialized_end=1524 _TRANSPORTCONFIG._serialized_start=96 _TRANSPORTCONFIG._serialized_end=254 _STREAMCONFIG._serialized_start=257 diff --git a/xray_api/proto/transport/internet/splithttp/config_pb2.py b/xray_api/proto/transport/internet/splithttp/config_pb2.py new file mode 100644 index 000000000..56e9ae4cf --- /dev/null +++ b/xray_api/proto/transport/internet/splithttp/config_pb2.py @@ -0,0 +1,48 @@ +# -*- coding: utf-8 -*- +# Generated by the protocol buffer compiler. DO NOT EDIT! +# source: transport/internet/splithttp/config.proto +"""Generated protocol buffer code.""" +from google.protobuf import descriptor as _descriptor +from google.protobuf import descriptor_pool as _descriptor_pool +from google.protobuf import message as _message +from google.protobuf import reflection as _reflection +from google.protobuf import symbol_database as _symbol_database +# @@protoc_insertion_point(imports) + +_sym_db = _symbol_database.Default() + + + + +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n)transport/internet/splithttp/config.proto\x12!xray.transport.internet.splithttp\"\xcf\x01\n\x06\x43onfig\x12\x0c\n\x04host\x18\x01 \x01(\t\x12\x0c\n\x04path\x18\x02 \x01(\t\x12\x45\n\x06header\x18\x03 \x03(\x0b\x32\x35.xray.transport.internet.splithttp.Config.HeaderEntry\x12\x1c\n\x14maxConcurrentUploads\x18\x04 \x01(\x05\x12\x15\n\rmaxUploadSize\x18\x05 \x01(\x05\x1a-\n\x0bHeaderEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x01(\t:\x02\x38\x01\x42\x85\x01\n%com.xray.transport.internet.splithttpP\x01Z6github.com/xtls/xray-core/transport/internet/splithttp\xaa\x02!Xray.Transport.Internet.SplitHttpb\x06proto3') + + + +_CONFIG = DESCRIPTOR.message_types_by_name['Config'] +_CONFIG_HEADERENTRY = _CONFIG.nested_types_by_name['HeaderEntry'] +Config = _reflection.GeneratedProtocolMessageType('Config', (_message.Message,), { + + 'HeaderEntry' : _reflection.GeneratedProtocolMessageType('HeaderEntry', (_message.Message,), { + 'DESCRIPTOR' : _CONFIG_HEADERENTRY, + '__module__' : 'transport.internet.splithttp.config_pb2' + # @@protoc_insertion_point(class_scope:xray.transport.internet.splithttp.Config.HeaderEntry) + }) + , + 'DESCRIPTOR' : _CONFIG, + '__module__' : 'transport.internet.splithttp.config_pb2' + # @@protoc_insertion_point(class_scope:xray.transport.internet.splithttp.Config) + }) +_sym_db.RegisterMessage(Config) +_sym_db.RegisterMessage(Config.HeaderEntry) + +if _descriptor._USE_C_DESCRIPTORS == False: + + DESCRIPTOR._options = None + DESCRIPTOR._serialized_options = b'\n%com.xray.transport.internet.splithttpP\001Z6github.com/xtls/xray-core/transport/internet/splithttp\252\002!Xray.Transport.Internet.SplitHttp' + _CONFIG_HEADERENTRY._options = None + _CONFIG_HEADERENTRY._serialized_options = b'8\001' + _CONFIG._serialized_start=81 + _CONFIG._serialized_end=288 + _CONFIG_HEADERENTRY._serialized_start=243 + _CONFIG_HEADERENTRY._serialized_end=288 +# @@protoc_insertion_point(module_scope) diff --git a/xray_api/proto/transport/internet/splithttp/config_pb2_grpc.py b/xray_api/proto/transport/internet/splithttp/config_pb2_grpc.py new file mode 100644 index 000000000..2daafffeb --- /dev/null +++ b/xray_api/proto/transport/internet/splithttp/config_pb2_grpc.py @@ -0,0 +1,4 @@ +# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT! +"""Client and server classes corresponding to protobuf-defined services.""" +import grpc + From 420edb8063ef9948b02ee05778ff5844af1a3dd8 Mon Sep 17 00:00:00 2001 From: Mohammad20000 Date: Wed, 3 Jul 2024 15:03:32 +0330 Subject: [PATCH 36/52] Add splithttp --- app/subscription/v2ray.py | 30 ++++++++++++++++++++++-------- app/xray/config.py | 5 +++++ 2 files changed, 27 insertions(+), 8 deletions(-) diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index 625432db5..41b547b74 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -1,15 +1,14 @@ import base64 import json -from random import choice import urllib.parse as urlparse +from random import choice from typing import Union -from uuid import UUID from urllib.parse import quote +from uuid import UUID from app.subscription.funcs import get_grpc_gun, get_grpc_multi from app.templates import render_template - -from config import (MUX_TEMPLATE, V2RAY_SUBSCRIPTION_TEMPLATE, USER_AGENT_TEMPLATE) +from config import MUX_TEMPLATE, USER_AGENT_TEMPLATE, V2RAY_SUBSCRIPTION_TEMPLATE class V2rayShareLink(str): @@ -114,9 +113,8 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): password=settings["password"], method=settings["method"], ) - + self.add_link(link=link) - @classmethod def vmess( @@ -162,7 +160,7 @@ def vmess( payload["sni"] = sni payload["fp"] = fp payload["alpn"] = alpn - if fs : + if fs: payload["fragment"] = fs if ais: payload["allowInsecure"] = 1 @@ -398,7 +396,6 @@ def reality_config(sni=None, fp=None, pbk=None, sid=None, spx=None): return realitySettings - def ws_config(self, path=None, host=None, random_user_agent=None): wsSettings = {} @@ -425,6 +422,19 @@ def httpupgrade_config(self, path=None, host=None, random_user_agent=None): return httpupgradeSettings + def splithttp_config(self, path=None, host=None, random_user_agent=None): + + splithttpSettings = {} + splithttpSettings["headers"] = {} + if path: + splithttpSettings["path"] = path + if host: + splithttpSettings["host"] = host + if random_user_agent: + splithttpSettings["headers"]["User-Agent"] = choice( + self.user_agent_list) + return splithttpSettings + @staticmethod def grpc_config(path=None, multiMode=False): @@ -558,6 +568,8 @@ def stream_setting_config(network=None, security=None, streamSettings["quicSettings"] = network_setting elif network == "httpupgrade": streamSettings["httpupgradeSettings"] = network_setting + elif network == "splithttp": + streamSettings["splithttpSettings"] = network_setting if sockopt: streamSettings['sockopt'] = sockopt @@ -683,6 +695,8 @@ def make_stream_setting(self, path=path, host=host, header=headers) elif net == "httpupgrade": network_setting = self.httpupgrade_config(path=path, host=host, random_user_agent=random_user_agent) + elif net == "splithttp": + network_setting = self.splithttp_config(path=path, host=host, random_user_agent=random_user_agent) if tls == "tls": tls_settings = self.tls_config(sni=sni, fp=fp, alpn=alpn, ais=ais) diff --git a/app/xray/config.py b/app/xray/config.py index e77b4682f..62ef93368 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -283,6 +283,11 @@ def _resolve_inbounds(self): host = net_settings.get('host', '') settings['host'] = [host] + elif net == 'splithttp': + settings['path'] = net_settings.get('path', '') + host = net_settings.get('host', '') + settings['host'] = [host] + else: settings['path'] = net_settings.get('path', '') host = net_settings.get( From 5bbd2c71cd811964dc2de941539621ced3336083 Mon Sep 17 00:00:00 2001 From: Erfan <143827987+erfjab@users.noreply.github.com> Date: Wed, 3 Jul 2024 15:33:30 +0330 Subject: [PATCH 37/52] change RANDOMIZE_SUBSCRIPTION_CONFIGS name .env.example --- .env.example | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.env.example b/.env.example index f5c48ed8a..9c4eb1191 100644 --- a/.env.example +++ b/.env.example @@ -49,7 +49,7 @@ UVICORN_PORT = 8000 # SUB_PROFILE_TITLE = "Susbcription" # SUB_SUPPORT_URL = "https://t.me/support" # SUB_UPDATE_INTERVAL = "12" -# RANDOMIZE_SUB_CONFIGS_STATUS = True +# RANDOMIZE_SUBSCRIPTION_CONFIGS = True # SQLALCHEMY_DATABASE_URL = "sqlite:///db.sqlite3" From f5c8f1cd78478f1a1fa7b423b650d0de9b9d41cd Mon Sep 17 00:00:00 2001 From: Erfan <143827987+erfjab@users.noreply.github.com> Date: Wed, 3 Jul 2024 15:33:58 +0330 Subject: [PATCH 38/52] change RANDOMIZE_SUBSCRIPTION_CONFIGS name config.py --- config.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config.py b/config.py index 6c1ee54bc..6902e1ba9 100755 --- a/config.py +++ b/config.py @@ -97,7 +97,7 @@ SUB_UPDATE_INTERVAL = config("SUB_UPDATE_INTERVAL", default="12") SUB_SUPPORT_URL = config("SUB_SUPPORT_URL", default="https://t.me/") SUB_PROFILE_TITLE = config("SUB_PROFILE_TITLE", default="Subscription") -RANDOM_SUB_CONFIGS_STATUS = config("RANDOM_SUB_CONFIGS_STATUS", default=False, cast=bool) +RANDOMIZE_SUBSCRIPTION_CONFIGS = config("RANDOMIZE_SUBSCRIPTION_CONFIGS", default=False, cast=bool) # discord webhook log DISCORD_WEBHOOK_URL = config("DISCORD_WEBHOOK_URL", default="") From 36c34e601da00de2e3269eb2e1b70f6f11a81719 Mon Sep 17 00:00:00 2001 From: Erfan <143827987+erfjab@users.noreply.github.com> Date: Wed, 3 Jul 2024 15:34:27 +0330 Subject: [PATCH 39/52] change RANDOMIZE_SUBSCRIPTION_CONFIGS name share.py --- app/subscription/share.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app/subscription/share.py b/app/subscription/share.py index c9eb9a520..a191e5620 100644 --- a/app/subscription/share.py +++ b/app/subscription/share.py @@ -19,7 +19,7 @@ from config import (ACTIVE_STATUS_TEXT, DISABLED_STATUS_TEXT, EXPIRED_STATUS_TEXT, LIMITED_STATUS_TEXT, - ONHOLD_STATUS_TEXT, RANDOM_SUB_CONFIGS_STATUS) + ONHOLD_STATUS_TEXT, RANDOMIZE_SUBSCRIPTION_CONFIGS) SERVER_IP = get_public_ip() SERVER_IPV6 = get_public_ipv6() @@ -157,7 +157,7 @@ def generate_subscription( else: raise ValueError(f'Unsupported format "{config_format}"') - if RANDOM_SUB_CONFIGS_STATUS is not False: + if RANDOMIZE_SUBSCRIPTION_CONFIGS is not False: config = randomize_sub_config(config, config_format) if as_base64: From 502d1e12b57312b16db8dacad792e0d2ce49a4c0 Mon Sep 17 00:00:00 2001 From: SaintShit Date: Wed, 3 Jul 2024 15:48:38 +0330 Subject: [PATCH 40/52] fix: parse user-agent for FlClash and Mihomo --- app/views/subscription.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/views/subscription.py b/app/views/subscription.py index 3c6fe4591..f176a6329 100644 --- a/app/views/subscription.py +++ b/app/views/subscription.py @@ -79,7 +79,7 @@ def get_subscription_user_info(user: UserResponse) -> dict: crud.update_user_sub(db, dbuser, user_agent) - if re.match('^([Cc]lash-verge|[Cc]lash[-\.]?[Mm]eta)', user_agent): + if re.match('^([Cc]lash-verge|[Cc]lash[-\.]?[Mm]eta|[Ff][Ll][Cc]lash|[Mm]ihomo)', user_agent): conf = generate_subscription(user=user, config_format="clash-meta", as_base64=False) return Response(content=conf, media_type="text/yaml", headers=response_headers) From a2ad3171eeebc8308027578eaedd956deff4caf8 Mon Sep 17 00:00:00 2001 From: SaintShit Date: Wed, 3 Jul 2024 16:56:28 +0330 Subject: [PATCH 41/52] fix: prevent error while only_full_group_by is enabled on sql --- app/xray/config.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/app/xray/config.py b/app/xray/config.py index e77b4682f..6e206f5e4 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -334,7 +334,9 @@ def include_db_users(self) -> XRayConfig: db_models.User.status.in_([UserStatus.active, UserStatus.on_hold]) ).group_by( func.lower(db_models.Proxy.type), - db_models.User.id + db_models.User.id, + db_models.User.username, + db_models.Proxy.settings, ) result = query.all() From 56a19c1eab05d7b0333dfeb9e2ea5017eecd88be Mon Sep 17 00:00:00 2001 From: Mohammad20000 Date: Wed, 3 Jul 2024 17:18:38 +0330 Subject: [PATCH 42/52] Add authority adn random UA to gRPC --- app/subscription/v2ray.py | 14 +++++++++----- app/xray/config.py | 3 ++- 2 files changed, 11 insertions(+), 6 deletions(-) diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index 41b547b74..c87715f97 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -218,6 +218,7 @@ def vless(cls, if net == 'grpc': payload['serviceName'] = path + payload["authority"] = host payload["host"] = host if multiMode: payload["mode"] = "multi" @@ -287,7 +288,7 @@ def trojan(cls, if net == 'grpc': payload['serviceName'] = path - payload["host"] = host + payload["authority"] = host if multiMode: payload["mode"] = "multi" else: @@ -435,18 +436,20 @@ def splithttp_config(self, path=None, host=None, random_user_agent=None): self.user_agent_list) return splithttpSettings - @staticmethod - def grpc_config(path=None, multiMode=False): + def grpc_config(self, path=None, host=None, multiMode=False, random_user_agent=None): grpcSettings = {} if path: grpcSettings["serviceName"] = path + if host: + grpcSettings["authority"] = host grpcSettings["multiMode"] = multiMode grpcSettings["idle_timeout"] = 60 grpcSettings["health_check_timeout"] = 20 grpcSettings["permit_without_stream"] = False grpcSettings["initial_windows_size"] = 0 - + if random_user_agent: + grpcSettings["user_agent"] = choice(self.user_agent_list) return grpcSettings def tcp_http_config(self, path=None, host=None, random_user_agent=None): @@ -682,7 +685,8 @@ def make_stream_setting(self, if net == "ws": network_setting = self.ws_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "grpc": - network_setting = self.grpc_config(path=path, multiMode=multiMode) + network_setting = self.grpc_config(path=path, host=host, multiMode=multiMode, + random_user_agent=random_user_agent) elif net == "h2": network_setting = self.h2_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "kcp": diff --git a/app/xray/config.py b/app/xray/config.py index 62ef93368..311cd2187 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -270,7 +270,8 @@ def _resolve_inbounds(self): elif net == 'grpc' or net == 'gun': settings['header_type'] = '' settings['path'] = net_settings.get('serviceName', '') - settings['host'] = [] + host = net_settings.get('authority', '') + settings['host'] = [host] settings['multiMode'] = net_settings.get('multiMode', False) elif net == 'quic': From da2ec7ce51d1ea5b2c75a78d7cc1320b08968a3d Mon Sep 17 00:00:00 2001 From: Mohammad20000 Date: Wed, 3 Jul 2024 17:29:26 +0330 Subject: [PATCH 43/52] fix authority for vless --- app/subscription/v2ray.py | 1 - 1 file changed, 1 deletion(-) diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index c87715f97..a2a5e28f6 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -219,7 +219,6 @@ def vless(cls, if net == 'grpc': payload['serviceName'] = path payload["authority"] = host - payload["host"] = host if multiMode: payload["mode"] = "multi" else: From 475dc6ec1f69952ad635949a7db69d6889e95401 Mon Sep 17 00:00:00 2001 From: SaintShit Date: Wed, 3 Jul 2024 17:36:22 +0330 Subject: [PATCH 44/52] fix: use utils.readable_size to show admin usages in cli --- cli/admin.py | 23 ++++++++++++----------- 1 file changed, 12 insertions(+), 11 deletions(-) diff --git a/cli/admin.py b/cli/admin.py index de6c81dd6..ddbf7c6cf 100644 --- a/cli/admin.py +++ b/cli/admin.py @@ -1,17 +1,18 @@ from typing import Optional, Union import typer -from rich.table import Table +from decouple import UndefinedValueError, config from rich.console import Console from rich.panel import Panel -from sqlalchemy.exc import IntegrityError +from rich.table import Table from sqlalchemy import func -from decouple import config, UndefinedValueError +from sqlalchemy.exc import IntegrityError -from app.db import GetDB -from app.db import crud +from app.db import GetDB, crud from app.db.models import Admin, User from app.models.admin import AdminCreate, AdminPartialModify +from app.utils.system import readable_size + from . import utils app = typer.Typer(no_args_is_help=True) @@ -35,16 +36,16 @@ def validate_discord_webhook(value: str) -> Union[str, None]: return value -def calculate_admin_usage(admin_id: int) -> int: +def calculate_admin_usage(admin_id: int) -> str: with GetDB() as db: usage = db.query(func.sum(User.used_traffic)).filter_by(admin_id=admin_id).first()[0] - return 0 if not usage else int(usage / 1024 / 1024 / 1024) # to GB + return readable_size(int(usage or 0)) -def calculate_admin_reseted_usage(admin_id: int) -> int: +def calculate_admin_reseted_usage(admin_id: int) -> str: with GetDB() as db: usage = db.query(func.sum(User.reseted_usage)).filter_by(admin_id=admin_id).first()[0] - return 0 if not usage else int(usage / 1024 / 1024 / 1024) # to GB + return readable_size(int(usage or 0)) @app.command(name="list") @@ -60,8 +61,8 @@ def list_admins( table=Table("Username", 'Usage', 'Reseted usage', "Is sudo", "Created at", "Telegram ID", "Discord Webhook"), rows=[ (str(admin.username), - f'{calculate_admin_usage(admin.id)}GB', - f'{calculate_admin_reseted_usage(admin.id)}GB', + calculate_admin_usage(admin.id), + calculate_admin_reseted_usage(admin.id), "✔️" if admin.is_sudo else "✖️", utils.readable_datetime(admin.created_at), str(admin.telegram_id or "✖️"), From cab20737788e93932646aca103d76ed063cde66d Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Wed, 3 Jul 2024 19:22:51 +0330 Subject: [PATCH 45/52] fix: splithttp and alpn --- app/subscription/share.py | 2 +- app/subscription/v2ray.py | 54 +++++++++++++++++++++++++++++++-------- app/xray/config.py | 2 ++ 3 files changed, 47 insertions(+), 11 deletions(-) diff --git a/app/subscription/share.py b/app/subscription/share.py index a191e5620..998aad8dc 100644 --- a/app/subscription/share.py +++ b/app/subscription/share.py @@ -314,7 +314,7 @@ def process_inbounds_and_tags( "sni": sni, "host": req_host, "tls": inbound["tls"] if host["tls"] is None else host["tls"], - "alpn": host["alpn"] or inbound.get("alpn", ""), + "alpn": host["alpn"].rsplit(sep=",") or inbound.get("alpn", ""), "path": path, "fp": host["fingerprint"] or inbound.get("fp", ""), "ais": host["allowinsecure"] diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index a2a5e28f6..ddeb7b20c 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -57,6 +57,8 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), multiMode=multi_mode, + max_upload_size=inbound.get('max_upload_size', 1), + max_concurrent_uploads=inbound.get('max_concurrent_uploads', 10), ) elif inbound["protocol"] == "vless": @@ -80,6 +82,8 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), multiMode=multi_mode, + max_upload_size=inbound.get('max_upload_size', 1), + max_concurrent_uploads=inbound.get('max_concurrent_uploads', 10), ) elif inbound["protocol"] == "trojan": @@ -103,6 +107,8 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), multiMode=multi_mode, + max_upload_size=inbound.get('max_upload_size', 1), + max_concurrent_uploads=inbound.get('max_concurrent_uploads', 10), ) elif inbound["protocol"] == "shadowsocks": @@ -137,6 +143,8 @@ def vmess( ais="", fs="", multiMode: bool = False, + max_upload_size: int = 1, + max_concurrent_uploads: int = 10, ): payload = { "add": address, @@ -178,6 +186,10 @@ def vmess( else: payload["mode"] = "gun" + elif net == "splithttp": + payload["maxUploadSize"] = max_upload_size + payload["maxConcurrentUploads"] = max_concurrent_uploads + return ( "vmess://" + base64.b64encode( @@ -206,7 +218,9 @@ def vless(cls, ais='', fs="", multiMode: bool = False, - ): + max_upload_size: int = 1, + max_concurrent_uploads: int = 10, + ): payload = { "security": tls, @@ -227,6 +241,11 @@ def vless(cls, elif net == 'quic': payload['key'] = path payload["quicSecurity"] = host + + elif net == "splithttp": + payload["maxUploadSize"] = max_upload_size + payload["maxConcurrentUploads"] = max_concurrent_uploads + else: payload["path"] = path payload["host"] = host @@ -275,6 +294,8 @@ def trojan(cls, ais='', fs="", multiMode: bool = False, + max_upload_size: int = 1, + max_concurrent_uploads: int = 10, ): payload = { @@ -293,6 +314,10 @@ def trojan(cls, else: payload["mode"] = "gun" + elif net == "splithttp": + payload["maxUploadSize"] = max_upload_size + payload["maxConcurrentUploads"] = max_concurrent_uploads + elif net == 'quic': payload['key'] = path payload["quicSecurity"] = host @@ -422,7 +447,10 @@ def httpupgrade_config(self, path=None, host=None, random_user_agent=None): return httpupgradeSettings - def splithttp_config(self, path=None, host=None, random_user_agent=None): + def splithttp_config(self, path=None, host=None, random_user_agent=None, + max_upload_size: int = 1, + max_concurrent_uploads: int = 10, + ): splithttpSettings = {} splithttpSettings["headers"] = {} @@ -433,9 +461,12 @@ def splithttp_config(self, path=None, host=None, random_user_agent=None): if random_user_agent: splithttpSettings["headers"]["User-Agent"] = choice( self.user_agent_list) + splithttpSettings["maxUploadSize"] = max_upload_size + splithttpSettings["maxConcurrentUploads"] = max_concurrent_uploads + return splithttpSettings - def grpc_config(self, path=None, host=None, multiMode=False, random_user_agent=None): + def grpc_config(self, path=None, host=None, multiMode=False): grpcSettings = {} if path: @@ -447,8 +478,7 @@ def grpc_config(self, path=None, host=None, multiMode=False, random_user_agent=N grpcSettings["health_check_timeout"] = 20 grpcSettings["permit_without_stream"] = False grpcSettings["initial_windows_size"] = 0 - if random_user_agent: - grpcSettings["user_agent"] = choice(self.user_agent_list) + return grpcSettings def tcp_http_config(self, path=None, host=None, random_user_agent=None): @@ -679,13 +709,14 @@ def make_stream_setting(self, dialer_proxy='', multiMode: bool = False, random_user_agent: bool = False, + max_upload_size: int = 1, + max_concurrent_uploads: int = 10, ): if net == "ws": network_setting = self.ws_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "grpc": - network_setting = self.grpc_config(path=path, host=host, multiMode=multiMode, - random_user_agent=random_user_agent) + network_setting = self.grpc_config(path=path, host=host, multiMode=multiMode) elif net == "h2": network_setting = self.h2_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "kcp": @@ -694,12 +725,13 @@ def make_stream_setting(self, elif net == "tcp": network_setting = self.tcp_http_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "quic": - network_setting = self.quic_config( - path=path, host=host, header=headers) + network_setting = self.quic_config(path=path, host=host, header=headers) elif net == "httpupgrade": network_setting = self.httpupgrade_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "splithttp": - network_setting = self.splithttp_config(path=path, host=host, random_user_agent=random_user_agent) + network_setting = self.splithttp_config(path=path, host=host, random_user_agent=random_user_agent, + max_upload_size=max_upload_size, + max_concurrent_uploads=max_concurrent_uploads) if tls == "tls": tls_settings = self.tls_config(sni=sni, fp=fp, alpn=alpn, ais=ais) @@ -805,6 +837,8 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): dialer_proxy=dialer_proxy, multiMode=multi_mode, random_user_agent=inbound.get('random_user_agent', False), + max_upload_size=inbound.get('max_upload_size', 1), + max_concurrent_uploads=inbound.get('max_concurrent_uploads', 10), ) mux_json = json.loads(self.mux_template) diff --git a/app/xray/config.py b/app/xray/config.py index b99f40efe..2ac01dd0d 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -288,6 +288,8 @@ def _resolve_inbounds(self): settings['path'] = net_settings.get('path', '') host = net_settings.get('host', '') settings['host'] = [host] + settings['maxUploadSize'] = net_settings.get('maxUploadSize', 1) + settings['maxConcurrentUploads'] = net_settings.get('maxConcurrentUploads', 10) else: settings['path'] = net_settings.get('path', '') From e22bf9b0284f0a519c899bd08c010e62de50f07b Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Wed, 3 Jul 2024 19:37:26 +0330 Subject: [PATCH 46/52] fix: add h3 to alpn enum in db --- .../305943d779c4_add_h3_to_alpn_enum.py | 118 ++++++++++++++++++ 1 file changed, 118 insertions(+) create mode 100644 app/db/migrations/versions/305943d779c4_add_h3_to_alpn_enum.py diff --git a/app/db/migrations/versions/305943d779c4_add_h3_to_alpn_enum.py b/app/db/migrations/versions/305943d779c4_add_h3_to_alpn_enum.py new file mode 100644 index 000000000..8cea6c637 --- /dev/null +++ b/app/db/migrations/versions/305943d779c4_add_h3_to_alpn_enum.py @@ -0,0 +1,118 @@ +"""add h3 to alpn enum + +Revision ID: 305943d779c4 +Revises: 31f92220c0d0 +Create Date: 2024-07-03 19:27:15.282711 + +""" +from alembic import op +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision = '305943d779c4' +down_revision = '31f92220c0d0' +branch_labels = None +depends_on = None + + +# Describing of enum +enum_name = "alpn" +temp_enum_name = f"temp_{enum_name}" +old_values = ("", "h2", "http/1.1", "h2,http/1.1") +new_values = ("h3", "h3,h2", "h3,h2,http/1.1", *old_values) +# on downgrade convert [0] to [1] +downgrade_from = ("h3", "h3,h2", "h3,h2,http/1.1") +downgrade_to = ("") +old_type = sa.Enum(*old_values, name=enum_name) +new_type = sa.Enum(*new_values, name=enum_name) +temp_type = sa.Enum(*new_values, name=temp_enum_name) + + +# Describing of table +table_name = "hosts" +column_name = "alpn" +temp_table = sa.sql.table( + table_name, + sa.Column( + column_name, + new_type, + nullable=False + ) +) + + +def upgrade(): + # temp type to use instead of old one + temp_type.create(op.get_bind(), checkfirst=False) + + # changing of column type from old enum to new one. + # SQLite will create temp table for this + with op.batch_alter_table(table_name) as batch_op: + batch_op.alter_column( + column_name, + existing_type=old_type, + type_=temp_type, + existing_nullable=False, + postgresql_using=f"{column_name}::text::{temp_enum_name}" + ) + + # remove old enum, create new enum + old_type.drop(op.get_bind(), checkfirst=False) + new_type.create(op.get_bind(), checkfirst=False) + + # changing of column type from temp enum to new one. + # SQLite will create temp table for this + with op.batch_alter_table(table_name) as batch_op: + batch_op.alter_column( + column_name, + existing_type=temp_type, + type_=new_type, + existing_nullable=False, + postgresql_using=f"{column_name}::text::{enum_name}" + ) + + # remove temp enum + temp_type.drop(op.get_bind(), checkfirst=False) + + +def downgrade(): + # old enum don't have new value anymore. + # before downgrading from new enum to old one, + # we should replace new value from new enum with + # somewhat of old values from old enum + op.execute( + temp_table + .update() + .where( + temp_table.c.status in downgrade_from + ) + .values( + status=downgrade_to + ) + ) + + temp_type.create(op.get_bind(), checkfirst=False) + + with op.batch_alter_table(table_name) as batch_op: + batch_op.alter_column( + column_name, + existing_type=new_type, + type_=temp_type, + existing_nullable=False, + postgresql_using=f"{column_name}::text::{temp_enum_name}" + ) + + new_type.drop(op.get_bind(), checkfirst=False) + old_type.create(op.get_bind(), checkfirst=False) + + with op.batch_alter_table(table_name) as batch_op: + batch_op.alter_column( + column_name, + existing_type=temp_type, + type_=old_type, + existing_nullable=False, + postgresql_using=f"{column_name}::text::{enum_name}" + ) + + temp_type.drop(op.get_bind(), checkfirst=False) From 554233a2e0cabe9bd6380b38bc6aaa2d435664a0 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Wed, 3 Jul 2024 21:15:15 +0330 Subject: [PATCH 47/52] fix: maxUploadSize default value --- app/subscription/v2ray.py | 16 ++++++++-------- app/xray/config.py | 2 +- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index ddeb7b20c..ff5eb5c85 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -57,7 +57,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), multiMode=multi_mode, - max_upload_size=inbound.get('max_upload_size', 1), + max_upload_size=inbound.get('max_upload_size', 1000000), max_concurrent_uploads=inbound.get('max_concurrent_uploads', 10), ) @@ -82,7 +82,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), multiMode=multi_mode, - max_upload_size=inbound.get('max_upload_size', 1), + max_upload_size=inbound.get('max_upload_size', 1000000), max_concurrent_uploads=inbound.get('max_concurrent_uploads', 10), ) @@ -107,7 +107,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): ais=inbound.get("ais", ""), fs=inbound.get("fragment_setting", ""), multiMode=multi_mode, - max_upload_size=inbound.get('max_upload_size', 1), + max_upload_size=inbound.get('max_upload_size', 1000000), max_concurrent_uploads=inbound.get('max_concurrent_uploads', 10), ) @@ -143,7 +143,7 @@ def vmess( ais="", fs="", multiMode: bool = False, - max_upload_size: int = 1, + max_upload_size: int = 1000000, max_concurrent_uploads: int = 10, ): payload = { @@ -218,7 +218,7 @@ def vless(cls, ais='', fs="", multiMode: bool = False, - max_upload_size: int = 1, + max_upload_size: int = 1000000, max_concurrent_uploads: int = 10, ): @@ -294,7 +294,7 @@ def trojan(cls, ais='', fs="", multiMode: bool = False, - max_upload_size: int = 1, + max_upload_size: int = 1000000, max_concurrent_uploads: int = 10, ): @@ -448,7 +448,7 @@ def httpupgrade_config(self, path=None, host=None, random_user_agent=None): return httpupgradeSettings def splithttp_config(self, path=None, host=None, random_user_agent=None, - max_upload_size: int = 1, + max_upload_size: int = 1000000, max_concurrent_uploads: int = 10, ): @@ -837,7 +837,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): dialer_proxy=dialer_proxy, multiMode=multi_mode, random_user_agent=inbound.get('random_user_agent', False), - max_upload_size=inbound.get('max_upload_size', 1), + max_upload_size=inbound.get('max_upload_size', 1000000), max_concurrent_uploads=inbound.get('max_concurrent_uploads', 10), ) diff --git a/app/xray/config.py b/app/xray/config.py index 2ac01dd0d..fe1af2a78 100644 --- a/app/xray/config.py +++ b/app/xray/config.py @@ -288,7 +288,7 @@ def _resolve_inbounds(self): settings['path'] = net_settings.get('path', '') host = net_settings.get('host', '') settings['host'] = [host] - settings['maxUploadSize'] = net_settings.get('maxUploadSize', 1) + settings['maxUploadSize'] = net_settings.get('maxUploadSize', 1000000) settings['maxConcurrentUploads'] = net_settings.get('maxConcurrentUploads', 10) else: From 20739589aae75943ed023f6d1b5de091888598f1 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Thu, 4 Jul 2024 13:42:14 +0330 Subject: [PATCH 48/52] feat(subscription): random user agent for grpc --- app/subscription/clash.py | 20 +++++++++++++++++--- app/subscription/singbox.py | 18 +++++++++++++++++- app/subscription/v2ray.py | 26 +++++++++++++++++++++----- app/templates/user_agent/grpc.json | 20 ++++++++++++++++++++ config.py | 1 + 5 files changed, 76 insertions(+), 9 deletions(-) create mode 100644 app/templates/user_agent/grpc.json diff --git a/app/subscription/clash.py b/app/subscription/clash.py index 369273c11..c37f747ce 100644 --- a/app/subscription/clash.py +++ b/app/subscription/clash.py @@ -5,8 +5,12 @@ from app.subscription.funcs import get_grpc_gun from app.templates import render_template -from config import (CLASH_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE, - USER_AGENT_TEMPLATE) +from config import ( + CLASH_SUBSCRIPTION_TEMPLATE, + MUX_TEMPLATE, + USER_AGENT_TEMPLATE, + GRPC_USER_AGENT_TEMPLATE, +) class ClashConfiguration(object): @@ -27,6 +31,14 @@ def __init__(self): else: self.user_agent_list = [] + temp_grpc_user_agent_data = render_template(GRPC_USER_AGENT_TEMPLATE) + grpc_user_agent_data = json.loads(temp_grpc_user_agent_data) + + if 'list' in grpc_user_agent_data and isinstance(grpc_user_agent_data['list'], list): + self.grpc_user_agent_data = grpc_user_agent_data['list'] + else: + self.grpc_user_agent_data = [] + def render(self): return yaml.dump( yaml.load( @@ -143,9 +155,11 @@ def make_node(self, if max_early_data: net_opts['v2ray-http-upgrade-fast-open'] = True - if network == 'grpc': + if network == 'grpc' or network == 'gun': if path: net_opts['grpc-service-name'] = path + if random_user_agent: + net_opts['header'] = {"User-Agent": choice(self.user_agent_list)} if network == 'h2': if path: diff --git a/app/subscription/singbox.py b/app/subscription/singbox.py index d237287c6..203e389c0 100644 --- a/app/subscription/singbox.py +++ b/app/subscription/singbox.py @@ -3,7 +3,12 @@ from app.templates import render_template from app.subscription.funcs import get_grpc_gun -from config import SINGBOX_SUBSCRIPTION_TEMPLATE, MUX_TEMPLATE, USER_AGENT_TEMPLATE +from config import ( + SINGBOX_SUBSCRIPTION_TEMPLATE, + MUX_TEMPLATE, + USER_AGENT_TEMPLATE, + GRPC_USER_AGENT_TEMPLATE, +) class SingBoxConfiguration(str): @@ -20,6 +25,14 @@ def __init__(self): else: self.user_agent_list = [] + temp_grpc_user_agent_data = render_template(GRPC_USER_AGENT_TEMPLATE) + grpc_user_agent_data = json.loads(temp_grpc_user_agent_data) + + if 'list' in grpc_user_agent_data and isinstance(grpc_user_agent_data['list'], list): + self.grpc_user_agent_data = grpc_user_agent_data['list'] + else: + self.grpc_user_agent_data = [] + def add_outbound(self, outbound_data): self.config["outbounds"].append(outbound_data) @@ -130,6 +143,9 @@ def transport_config(self, transport_config['ping_timeout'] = ping_timeout if permit_without_stream: transport_config['permit_without_stream'] = permit_without_stream + if random_user_agent: + transport_config['headers'] = {} + transport_config['headers']['User-Agent'] = choice(self.grpc_user_agent_data) elif transport_type == "httpupgrade": transport_config['host'] = host diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index ff5eb5c85..1bbb44d5f 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -8,7 +8,12 @@ from app.subscription.funcs import get_grpc_gun, get_grpc_multi from app.templates import render_template -from config import MUX_TEMPLATE, USER_AGENT_TEMPLATE, V2RAY_SUBSCRIPTION_TEMPLATE +from config import ( + MUX_TEMPLATE, + USER_AGENT_TEMPLATE, + V2RAY_SUBSCRIPTION_TEMPLATE, + GRPC_USER_AGENT_TEMPLATE, +) class V2rayShareLink(str): @@ -367,12 +372,20 @@ def __init__(self): self.mux_template = render_template(MUX_TEMPLATE) temp_user_agent_data = render_template(USER_AGENT_TEMPLATE) user_agent_data = json.loads(temp_user_agent_data) - + if 'list' in user_agent_data and isinstance(user_agent_data['list'], list): self.user_agent_list = user_agent_data['list'] else: self.user_agent_list = [] + temp_grpc_user_agent_data = render_template(GRPC_USER_AGENT_TEMPLATE) + grpc_user_agent_data = json.loads(temp_grpc_user_agent_data) + + if 'list' in grpc_user_agent_data and isinstance(grpc_user_agent_data['list'], list): + self.grpc_user_agent_data = grpc_user_agent_data['list'] + else: + self.grpc_user_agent_data = [] + def add_config(self, remarks, outbounds): json_template = json.loads(self.template) json_template["remarks"] = remarks @@ -466,7 +479,7 @@ def splithttp_config(self, path=None, host=None, random_user_agent=None, return splithttpSettings - def grpc_config(self, path=None, host=None, multiMode=False): + def grpc_config(self, path=None, host=None, multiMode=False, random_user_agent=None): grpcSettings = {} if path: @@ -477,7 +490,10 @@ def grpc_config(self, path=None, host=None, multiMode=False): grpcSettings["idle_timeout"] = 60 grpcSettings["health_check_timeout"] = 20 grpcSettings["permit_without_stream"] = False - grpcSettings["initial_windows_size"] = 0 + grpcSettings["initial_windows_size"] = 35536 + + if random_user_agent: + grpcSettings["user_agent"] = choice(self.grpc_user_agent_data) return grpcSettings @@ -716,7 +732,7 @@ def make_stream_setting(self, if net == "ws": network_setting = self.ws_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "grpc": - network_setting = self.grpc_config(path=path, host=host, multiMode=multiMode) + network_setting = self.grpc_config(path=path, host=host, multiMode=multiMode, random_user_agent=random_user_agent) elif net == "h2": network_setting = self.h2_config(path=path, host=host, random_user_agent=random_user_agent) elif net == "kcp": diff --git a/app/templates/user_agent/grpc.json b/app/templates/user_agent/grpc.json new file mode 100644 index 000000000..821671ea4 --- /dev/null +++ b/app/templates/user_agent/grpc.json @@ -0,0 +1,20 @@ +{ + "list": [ + "grpc-dotnet/2.41.0 (.NET 6.0.1; CLR 6.0.1; net6.0; windows; x64)", + "grpc-dotnet/2.41.0 (.NET 6.0.0-preview.7.21377.19; CLR 6.0.0; net6.0; osx; x64)", + "grpc-dotnet/2.41.0 (Mono 6.12.0.140; CLR 4.0.30319; netstandard2.0; osx; x64)", + "grpc-dotnet/2.41.0 (.NET 6.0.0-rc.1.21380.1; CLR 6.0.0; net6.0; linux; arm64)", + "grpc-dotnet/2.41.0 (.NET 5.0.8; CLR 5.0.8; net5.0; linux; arm64)", + "grpc-dotnet/2.41.0 (.NET Core; CLR 3.1.4; netstandard2.1; linux; arm64)", + "grpc-dotnet/2.41.0 (.NET Framework; CLR 4.0.30319.42000; netstandard2.0; windows; x86)", + "grpc-dotnet/2.41.0 (.NET 6.0.0-rc.1.21380.1; CLR 6.0.0; net6.0; windows; x64)", + "grpc-python-asyncio/1.62.1 grpc-c/39.0.0 (linux; chttp2)", + "grpc-go/1.58.1", + "grpc-java-okhttp/1.55.1", + "grpc-node/1.7.1 grpc-c/1.7.1 (osx; chttp2)", + "grpc-node/1.24.2 grpc-c/8.0.0 (linux; chttp2; ganges)", + "grpc-c++/1.16.0 grpc-c/6.0.0 (linux; nghttp2; hw)", + "grpc-node/1.19.0 grpc-c/7.0.0 (linux; chttp2; gold)", + "grpc-ruby/1.62.0 grpc-c/39.0.0 (osx; chttp2)]" + ] +} \ No newline at end of file diff --git a/config.py b/config.py index 6902e1ba9..f8afc673a 100755 --- a/config.py +++ b/config.py @@ -51,6 +51,7 @@ MUX_TEMPLATE = config("MUX_TEMPLATE", default="mux/default.json") V2RAY_SUBSCRIPTION_TEMPLATE = config("V2RAY_SUBSCRIPTION_TEMPLATE", default="v2ray/default.json") USER_AGENT_TEMPLATE = config("USER_AGENT_TEMPLATE", default="user_agent/default.json") +GRPC_USER_AGENT_TEMPLATE = config("GRPC_USER_AGENT_TEMPLATE", default="user_agent/grpc.json") USE_CUSTOM_JSON_DEFAULT = config("USE_CUSTOM_JSON_DEFAULT", default=False, cast=bool) From f04a8f9b3e64e5532cef52e878e6f8f9a5ed1fca Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Thu, 4 Jul 2024 16:11:51 +0330 Subject: [PATCH 49/52] fix(subscription): alpn for clash meta and standard v2ray --- app/subscription/share.py | 2 +- app/subscription/singbox.py | 2 +- app/subscription/v2ray.py | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/app/subscription/share.py b/app/subscription/share.py index 998aad8dc..a191e5620 100644 --- a/app/subscription/share.py +++ b/app/subscription/share.py @@ -314,7 +314,7 @@ def process_inbounds_and_tags( "sni": sni, "host": req_host, "tls": inbound["tls"] if host["tls"] is None else host["tls"], - "alpn": host["alpn"].rsplit(sep=",") or inbound.get("alpn", ""), + "alpn": host["alpn"] or inbound.get("alpn", ""), "path": path, "fp": host["fingerprint"] or inbound.get("fp", ""), "ais": host["allowinsecure"] diff --git a/app/subscription/singbox.py b/app/subscription/singbox.py index 203e389c0..50780a0bf 100644 --- a/app/subscription/singbox.py +++ b/app/subscription/singbox.py @@ -249,7 +249,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): sni=inbound['sni'], host=inbound['host'], path=path, - alpn=inbound.get('alpn', ''), + alpn=inbound.get('alpn', '').rsplit(sep=","), fp=inbound.get('fp', ''), pbk=inbound.get('pbk', ''), sid=inbound.get('sid', ''), diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index 1bbb44d5f..40dee1295 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -843,7 +843,7 @@ def add(self, remark: str, address: str, inbound: dict, settings: dict): sni=inbound['sni'], host=inbound['host'], path=path, - alpn=inbound.get('alpn', ''), + alpn=inbound.get('alpn', '').rsplit(sep=","), fp=inbound.get('fp', ''), pbk=inbound.get('pbk', ''), sid=inbound.get('sid', ''), From fbc87ed54a6def23734d0871624e70faf5cabc21 Mon Sep 17 00:00:00 2001 From: Random Guy <50927468+M03ED@users.noreply.github.com> Date: Thu, 4 Jul 2024 16:49:15 +0330 Subject: [PATCH 50/52] fix(subscription): path and host for splithttp --- app/subscription/v2ray.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/app/subscription/v2ray.py b/app/subscription/v2ray.py index 40dee1295..60d97213a 100644 --- a/app/subscription/v2ray.py +++ b/app/subscription/v2ray.py @@ -248,6 +248,8 @@ def vless(cls, payload["quicSecurity"] = host elif net == "splithttp": + payload["path"] = path + payload["host"] = host payload["maxUploadSize"] = max_upload_size payload["maxConcurrentUploads"] = max_concurrent_uploads @@ -320,6 +322,8 @@ def trojan(cls, payload["mode"] = "gun" elif net == "splithttp": + payload["path"] = path + payload["host"] = host payload["maxUploadSize"] = max_upload_size payload["maxConcurrentUploads"] = max_concurrent_uploads From dcc2dfa623446f85dd80899c04c4d60b02893ba7 Mon Sep 17 00:00:00 2001 From: SaintShit Date: Thu, 4 Jul 2024 17:39:39 +0330 Subject: [PATCH 51/52] fix: migration script 305943d779c4 --- .../305943d779c4_add_h3_to_alpn_enum.py | 19 ++++++++----------- 1 file changed, 8 insertions(+), 11 deletions(-) diff --git a/app/db/migrations/versions/305943d779c4_add_h3_to_alpn_enum.py b/app/db/migrations/versions/305943d779c4_add_h3_to_alpn_enum.py index 8cea6c637..6ff6d48e9 100644 --- a/app/db/migrations/versions/305943d779c4_add_h3_to_alpn_enum.py +++ b/app/db/migrations/versions/305943d779c4_add_h3_to_alpn_enum.py @@ -19,11 +19,11 @@ # Describing of enum enum_name = "alpn" temp_enum_name = f"temp_{enum_name}" -old_values = ("", "h2", "http/1.1", "h2,http/1.1") +old_values = ("none", "h2", "http/1.1", "h2,http/1.1") new_values = ("h3", "h3,h2", "h3,h2,http/1.1", *old_values) -# on downgrade convert [0] to [1] -downgrade_from = ("h3", "h3,h2", "h3,h2,http/1.1") -downgrade_to = ("") +# on downgrade +downgrade_from = ("h3", "h3,h2", "h3,h2,http/1.1", "") +downgrade_to = "none" old_type = sa.Enum(*old_values, name=enum_name) new_type = sa.Enum(*new_values, name=enum_name) temp_type = sa.Enum(*new_values, name=temp_enum_name) @@ -81,16 +81,13 @@ def downgrade(): # before downgrading from new enum to old one, # we should replace new value from new enum with # somewhat of old values from old enum - op.execute( + update_query = ( temp_table .update() - .where( - temp_table.c.status in downgrade_from - ) - .values( - status=downgrade_to - ) + .where(temp_table.c.alpn.in_(downgrade_from)) + .values(alpn=downgrade_to) ) + op.execute(update_query) temp_type.create(op.get_bind(), checkfirst=False) From 6dc0d38a0e58dcd60a413f0fe5af6597f6960fa4 Mon Sep 17 00:00:00 2001 From: SaintShit Date: Fri, 5 Jul 2024 11:32:21 +0330 Subject: [PATCH 52/52] fix: sing-box configuration doesn't support string port --- app/subscription/singbox.py | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/app/subscription/singbox.py b/app/subscription/singbox.py index 50780a0bf..efe72f9ca 100644 --- a/app/subscription/singbox.py +++ b/app/subscription/singbox.py @@ -4,8 +4,8 @@ from app.subscription.funcs import get_grpc_gun from config import ( - SINGBOX_SUBSCRIPTION_TEMPLATE, - MUX_TEMPLATE, + SINGBOX_SUBSCRIPTION_TEMPLATE, + MUX_TEMPLATE, USER_AGENT_TEMPLATE, GRPC_USER_AGENT_TEMPLATE, ) @@ -178,6 +178,10 @@ def make_outbound(self, random_user_agent: bool = False, ): + if isinstance(port, str): + ports = port.split(',') + port = int(choice(ports)) + config = { "type": type, "tag": remark,