| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165 | <?phpnamespace App\Http\Controllers\Client\Protocols;use Symfony\Component\Yaml\Yaml;class Stash{    public $flag = 'stash';    private $servers;    private $user;    public function __construct($user, $servers)    {        $this->user = $user;        $this->servers = $servers;    }    public function handle()    {        $servers = $this->servers;        $user = $this->user;        $appName = config('v2board.app_name', 'V2Board');        header("subscription-userinfo: upload={$user['u']}; download={$user['d']}; total={$user['transfer_enable']}; expire={$user['expired_at']}");        header('profile-update-interval: 24');        header("content-disposition: filename={$appName}");        // 暂时使用clash配置文件,后续根据Stash更新情况更新        $defaultConfig = base_path() . '/resources/rules/default.clash.yaml';        $customConfig = base_path() . '/resources/rules/custom.clash.yaml';        if (\File::exists($customConfig)) {            $config = Yaml::parseFile($customConfig);        } else {            $config = Yaml::parseFile($defaultConfig);        }        $proxy = [];        $proxies = [];        foreach ($servers as $item) {            if ($item['type'] === 'shadowsocks') {                array_push($proxy, self::buildShadowsocks($user['uuid'], $item));                array_push($proxies, $item['name']);            }            if ($item['type'] === 'v2ray') {                array_push($proxy, self::buildVmess($user['uuid'], $item));                array_push($proxies, $item['name']);            }            if ($item['type'] === 'trojan') {                array_push($proxy, self::buildTrojan($user['uuid'], $item));                array_push($proxies, $item['name']);            }        }        $config['proxies'] = array_merge($config['proxies'] ? $config['proxies'] : [], $proxy);        foreach ($config['proxy-groups'] as $k => $v) {            if (!is_array($config['proxy-groups'][$k]['proxies'])) continue;            $isFilter = false;            foreach ($config['proxy-groups'][$k]['proxies'] as $src) {                foreach ($proxies as $dst) {                    if ($this->isMatch($src, $dst)) {                        $isFilter = true;                        $config['proxy-groups'][$k]['proxies'] = array_diff($config['proxy-groups'][$k]['proxies'], [$src]);                        array_push($config['proxy-groups'][$k]['proxies'], $dst);                    }                }            }            if (!$isFilter) {                $config['proxy-groups'][$k]['proxies'] = array_merge($config['proxy-groups'][$k]['proxies'], $proxies);            }        }        // Force the current subscription domain to be a direct rule        $subsDomain = $_SERVER['SERVER_NAME'];        $subsDomainRule = "DOMAIN,{$subsDomain},DIRECT";        array_unshift($config['rules'], $subsDomainRule);        $yaml = Yaml::dump($config);        $yaml = str_replace('$app_name', config('v2board.app_name', 'V2Board'), $yaml);        return $yaml;    }    public static function buildShadowsocks($uuid, $server)    {        $array = [];        $array['name'] = $server['name'];        $array['type'] = 'ss';        $array['server'] = $server['host'];        $array['port'] = $server['port'];        $array['cipher'] = $server['cipher'];        $array['password'] = $uuid;        $array['udp'] = true;        return $array;    }    public static function buildVmess($uuid, $server)    {        $array = [];        $array['name'] = $server['name'];        $array['type'] = 'vmess';        $array['server'] = $server['host'];        $array['port'] = $server['port'];        $array['uuid'] = $uuid;        $array['alterId'] = 0;        $array['cipher'] = 'auto';        $array['udp'] = true;        if ($server['tls']) {            $array['tls'] = true;            if ($server['tlsSettings']) {                $tlsSettings = $server['tlsSettings'];                if (isset($tlsSettings['allowInsecure']) && !empty($tlsSettings['allowInsecure']))                    $array['skip-cert-verify'] = ($tlsSettings['allowInsecure'] ? true : false);                if (isset($tlsSettings['serverName']) && !empty($tlsSettings['serverName']))                    $array['servername'] = $tlsSettings['serverName'];            }        }        if ($server['network'] === 'ws') {            $array['network'] = 'ws';            if ($server['networkSettings']) {                $wsSettings = $server['networkSettings'];                $array['ws-opts'] = [];                if (isset($wsSettings['path']) && !empty($wsSettings['path']))                    $array['ws-opts']['path'] = $wsSettings['path'];                if (isset($wsSettings['headers']['Host']) && !empty($wsSettings['headers']['Host']))                    $array['ws-opts']['headers'] = ['Host' => $wsSettings['headers']['Host']];                // TODO: 2022.06.01 remove it                if (isset($wsSettings['path']) && !empty($wsSettings['path']))                    $array['ws-path'] = $wsSettings['path'];                if (isset($wsSettings['headers']['Host']) && !empty($wsSettings['headers']['Host']))                    $array['ws-headers'] = ['Host' => $wsSettings['headers']['Host']];            }        }        if ($server['network'] === 'grpc') {            $array['network'] = 'grpc';            if ($server['networkSettings']) {                $grpcSettings = $server['networkSettings'];                $array['grpc-opts'] = [];                $array['grpc-opts']['grpc-service-name'] = $grpcSettings['serviceName'];            }        }        return $array;    }    public static function buildTrojan($password, $server)    {        $array = [];        $array['name'] = $server['name'];        $array['type'] = 'trojan';        $array['server'] = $server['host'];        $array['port'] = $server['port'];        $array['password'] = $password;        $array['udp'] = true;        if (!empty($server['server_name'])) $array['sni'] = $server['server_name'];        if (!empty($server['allow_insecure'])) $array['skip-cert-verify'] = ($server['allow_insecure'] ? true : false);        return $array;    }    private function isMatch($exp, $str)    {        try {            return preg_match($exp, $str);        } catch (\Exception $e) {            return false;        }    }}
 |