Xboard/app/Protocols/General.php

219 lines
8.4 KiB
PHP
Raw Normal View History

2023-11-17 01:44:01 -05:00
<?php
namespace App\Protocols;
2025-01-06 12:20:11 -05:00
use App\Contracts\ProtocolInterface;
2023-11-17 01:44:01 -05:00
use App\Utils\Helper;
2025-01-06 12:20:11 -05:00
class General implements ProtocolInterface
2023-11-17 01:44:01 -05:00
{
2025-01-06 12:20:11 -05:00
public $flags = ['general', 'v2rayn', 'v2rayng', 'passwall', 'ssrplus', 'sagernet'];
2023-11-17 01:44:01 -05:00
private $servers;
private $user;
public function __construct($user, $servers)
{
$this->user = $user;
$this->servers = $servers;
}
2025-01-06 12:20:11 -05:00
public function getFlags(): array
{
return $this->flags;
}
2023-11-17 01:44:01 -05:00
public function handle()
{
$servers = $this->servers;
$user = $this->user;
$uri = '';
foreach ($servers as $item) {
if ($item['type'] === 'vmess') {
$uri .= self::buildVmess($user['uuid'], $item);
}
if ($item['type'] === 'vless') {
$uri .= self::buildVless($user['uuid'], $item);
}
if ($item['type'] === 'shadowsocks') {
$uri .= self::buildShadowsocks($item['password'], $item);
2023-11-17 01:44:01 -05:00
}
if ($item['type'] === 'trojan') {
$uri .= self::buildTrojan($user['uuid'], $item);
}
if ($item['type'] === 'hysteria') {
$uri .= self::buildHysteria($user['uuid'], $item);
}
2023-11-17 01:44:01 -05:00
}
return base64_encode($uri);
}
public static function buildShadowsocks($password, $server)
{
2025-01-06 12:20:11 -05:00
$protocol_settings = $server['protocol_settings'];
2023-11-17 01:44:01 -05:00
$name = rawurlencode($server['name']);
2025-01-09 02:58:32 -05:00
$password = data_get($server, 'password', $password);
2023-11-17 01:44:01 -05:00
$str = str_replace(
['+', '/', '='],
['-', '_', ''],
2025-01-06 12:20:11 -05:00
base64_encode("{$protocol_settings['cipher']}:{$password}")
2023-11-17 01:44:01 -05:00
);
return "ss://{$str}@{$server['host']}:{$server['port']}#{$name}\r\n";
}
public static function buildVmess($uuid, $server)
{
2025-01-06 12:20:11 -05:00
$protocol_settings = $server['protocol_settings'];
2023-11-17 01:44:01 -05:00
$config = [
"v" => "2",
"ps" => $server['name'],
"add" => $server['host'],
2025-01-06 12:20:11 -05:00
"port" => (string) $server['port'],
2023-11-17 01:44:01 -05:00
"id" => $uuid,
"aid" => '0',
2025-01-06 12:20:11 -05:00
"net" => $server['protocol_settings']['network'],
2023-11-17 01:44:01 -05:00
"type" => "none",
"host" => "",
"path" => "",
2025-01-06 12:20:11 -05:00
"tls" => $protocol_settings['tls'] ? "tls" : "",
"sni" => data_get($protocol_settings, 'tls_settings.server_name'),
2023-11-17 01:44:01 -05:00
];
2025-01-06 12:20:11 -05:00
switch ($protocol_settings['network']) {
case 'tcp':
2025-01-12 10:16:55 -05:00
if (data_get($protocol_settings, 'network_settings.header.type', 'none') !== 'none') {
$config['type'] = data_get($protocol_settings, 'network_settings.header.type', 'http');
$config['path'] = \Arr::random(data_get($protocol_settings, 'network_settings.header.request.path', ['/']));
$config['host'] = data_get($protocol_settings, 'network_settings.headers.Host') ? \Arr::random(data_get($protocol_settings, 'network_settings.headers.Host'), ['/']) : null;
}
2025-01-06 12:20:11 -05:00
break;
case 'ws':
$config['type'] = 'ws';
$config['path'] = data_get($protocol_settings, 'network_settings.path');
2025-01-12 08:10:52 -05:00
if ($host = data_get($protocol_settings, 'network_settings.headers.Host')) {
$config['host'] = $host;
}
2025-01-06 12:20:11 -05:00
break;
case 'grpc':
$config['type'] = 'grpc';
2025-01-09 07:02:19 -05:00
$config['path'] = data_get($protocol_settings, 'network_settings.serviceName');
2025-01-06 12:20:11 -05:00
break;
default:
break;
2023-11-17 01:44:01 -05:00
}
return "vmess://" . base64_encode(json_encode($config)) . "\r\n";
}
2025-01-06 12:20:11 -05:00
public static function buildVless($uuid, $server)
{
$protocol_settings = $server['protocol_settings'];
2023-11-17 01:44:01 -05:00
$host = $server['host']; //节点地址
$port = $server['port']; //节点端口
$name = $server['name']; //节点名称
$config = [
'mode' => 'multi', //grpc传输模式
'security' => '', //传输层安全 tls/reality
'encryption' => 'none', //加密方式
2025-01-06 12:20:11 -05:00
'type' => $server['protocol_settings']['network'], //传输协议
'flow' => $protocol_settings['flow'] ? $protocol_settings['flow'] : null,
2023-11-17 01:44:01 -05:00
];
2025-01-06 12:20:11 -05:00
// 处理TLS
switch ($server['protocol_settings']['tls']) {
case 1:
$config['security'] = "tls";
$config['sni'] = data_get($protocol_settings, 'tls_settings.server_name');
break;
case 2: //reality
$config['security'] = "reality";
$config['pbk'] = data_get($protocol_settings, 'reality_settings.public_key');
$config['sid'] = data_get($protocol_settings, 'reality_settings.short_id');
$config['sni'] = data_get($protocol_settings, 'reality_settings.server_name');
$config['servername'] = data_get($protocol_settings, 'reality_settings.server_name');
$config['spx'] = "/";
$config['fp'] = Helper::getRandFingerprint();
break;
default:
break;
2023-11-17 01:44:01 -05:00
}
2025-01-06 12:20:11 -05:00
// 处理传输协议
switch ($server['protocol_settings']['network']) {
case 'ws':
$config['path'] = data_get($protocol_settings, 'network_settings.path');
2025-01-12 08:10:52 -05:00
if ($host = data_get($protocol_settings, 'network_settings.headers.Host')) {
$config['host'] = $host;
}
2025-01-06 12:20:11 -05:00
break;
case 'grpc':
$config['serviceName'] = data_get($protocol_settings, 'network_settings.serviceName');
break;
case 'kcp':
$config['path'] = data_get($protocol_settings, 'network_settings.seed');
$config['type'] = data_get($protocol_settings, 'network_settings.header.type', 'none');
break;
case 'httpupgrade':
$config['path'] = data_get($protocol_settings, 'network_settings.path');
2025-01-12 08:10:52 -05:00
$config['host'] = data_get($protocol_settings, 'network_settings.headers.Host', $server['host']);
break;
case 'xhttp':
$config['path'] = data_get($protocol_settings, 'network_settings.path');
2025-01-12 08:10:52 -05:00
$config['host'] = data_get($protocol_settings, 'network_settings.headers.Host', $server['host']);
$config['mode'] = data_get($protocol_settings, 'network_settings.mode', 'auto');
$config['extra'] = data_get($protocol_settings, 'network_settings.extra') ? Helper::encodeURIComponent(data_get($protocol_settings, 'network_settings.extra')) : null;
break;
2023-11-17 01:44:01 -05:00
}
$user = $uuid . '@' . $host . ':' . $port;
$query = http_build_query($config);
$fragment = urlencode($name);
$link = sprintf("vless://%s?%s#%s\r\n", $user, $query, $fragment);
return $link;
}
public static function buildTrojan($password, $server)
{
2025-01-06 12:20:11 -05:00
$protocol_settings = $server['protocol_settings'];
2023-11-17 01:44:01 -05:00
$name = rawurlencode($server['name']);
$query = http_build_query([
2025-01-06 12:20:11 -05:00
'allowInsecure' => $protocol_settings['allow_insecure'],
'peer' => $protocol_settings['server_name'],
'sni' => $protocol_settings['server_name']
2023-11-17 01:44:01 -05:00
]);
$uri = "trojan://{$password}@{$server['host']}:{$server['port']}?{$query}#{$name}";
$uri .= "\r\n";
return $uri;
}
public static function buildHysteria($password, $server)
{
2025-01-06 12:20:11 -05:00
$protocol_settings = $server['protocol_settings'];
$params = [];
// Return empty if version is not 2
2025-01-06 12:20:11 -05:00
if ($server['protocol_settings']['version'] !== 2) {
return '';
}
2025-01-06 12:20:11 -05:00
if (data_get($protocol_settings, 'tls.server_name')) {
$params['sni'] = data_get($protocol_settings, 'tls.server_name');
$params['security'] = 'tls';
}
2025-01-06 12:20:11 -05:00
if (data_get($protocol_settings, 'obfs.open')) {
$params['obfs'] = 'salamander';
2025-01-06 12:20:11 -05:00
$params['obfs-password'] = data_get($protocol_settings, 'obfs.password');
}
2025-01-06 12:20:11 -05:00
$params['insecure'] = data_get($protocol_settings, 'tls.allow_insecure');
$query = http_build_query($params);
$name = rawurlencode($server['name']);
$uri = "hysteria2://{$password}@{$server['host']}:{$server['port']}?{$query}#{$name}";
$uri .= "\r\n";
return $uri;
}
2023-11-17 01:44:01 -05:00
}