2023-11-17 01:44:01 -05:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Http\Controllers\V1\Server;
|
|
|
|
|
2023-12-04 07:40:49 -05:00
|
|
|
use App\Exceptions\ApiException;
|
2023-11-17 01:44:01 -05:00
|
|
|
use App\Http\Controllers\Controller;
|
|
|
|
use App\Models\ServerVmess;
|
|
|
|
use App\Services\ServerService;
|
|
|
|
use App\Services\UserService;
|
|
|
|
use App\Utils\CacheKey;
|
|
|
|
use Illuminate\Http\Request;
|
|
|
|
use Illuminate\Support\Facades\Cache;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* V2ray Aurora
|
|
|
|
* Github: https://github.com/tokumeikoi/aurora
|
|
|
|
*/
|
|
|
|
class DeepbworkController extends Controller
|
|
|
|
{
|
|
|
|
CONST V2RAY_CONFIG = '{"log":{"loglevel":"debug","access":"access.log","error":"error.log"},"api":{"services":["HandlerService","StatsService"],"tag":"api"},"dns":{},"stats":{},"inbounds":[{"port":443,"protocol":"vmess","settings":{"clients":[]},"sniffing":{"enabled":true,"destOverride":["http","tls"]},"streamSettings":{"network":"tcp"},"tag":"proxy"},{"listen":"127.0.0.1","port":23333,"protocol":"dokodemo-door","settings":{"address":"0.0.0.0"},"tag":"api"}],"outbounds":[{"protocol":"freedom","settings":{}},{"protocol":"blackhole","settings":{},"tag":"block"}],"routing":{"rules":[{"type":"field","inboundTag":"api","outboundTag":"api"}]},"policy":{"levels":{"0":{"handshake":4,"connIdle":300,"uplinkOnly":5,"downlinkOnly":30,"statsUserUplink":true,"statsUserDownlink":true}}}}';
|
|
|
|
|
|
|
|
// 后端获取用户
|
|
|
|
public function user(Request $request)
|
|
|
|
{
|
|
|
|
ini_set('memory_limit', -1);
|
|
|
|
$nodeId = $request->input('node_id');
|
|
|
|
$server = ServerVmess::find($nodeId);
|
|
|
|
if (!$server) {
|
2023-12-06 15:01:32 -05:00
|
|
|
return $this->fail([400,'节点不存在']);
|
2023-11-17 01:44:01 -05:00
|
|
|
}
|
|
|
|
Cache::put(CacheKey::get('SERVER_VMESS_LAST_CHECK_AT', $server->id), time(), 3600);
|
2024-04-09 12:51:03 -04:00
|
|
|
$users = ServerService::getAvailableUsers($server->group_id);
|
2023-11-17 01:44:01 -05:00
|
|
|
$result = [];
|
|
|
|
foreach ($users as $user) {
|
|
|
|
$user->v2ray_user = [
|
|
|
|
"uuid" => $user->uuid,
|
|
|
|
"email" => sprintf("%s@v2board.user", $user->uuid),
|
|
|
|
"alter_id" => 0,
|
|
|
|
"level" => 0,
|
|
|
|
];
|
2023-12-15 19:05:09 -05:00
|
|
|
unset($user->uuid);
|
2023-11-17 01:44:01 -05:00
|
|
|
array_push($result, $user);
|
|
|
|
}
|
|
|
|
$eTag = sha1(json_encode($result));
|
|
|
|
if (strpos($request->header('If-None-Match'), $eTag) !== false ) {
|
|
|
|
return response(null,304);
|
|
|
|
}
|
|
|
|
return response([
|
|
|
|
'msg' => 'ok',
|
|
|
|
'data' => $result,
|
|
|
|
])->header('ETag', "\"{$eTag}\"");
|
|
|
|
}
|
|
|
|
|
|
|
|
// 后端提交数据
|
|
|
|
public function submit(Request $request)
|
|
|
|
{
|
|
|
|
$server = ServerVmess::find($request->input('node_id'));
|
|
|
|
if (!$server) {
|
|
|
|
return response([
|
|
|
|
'ret' => 0,
|
|
|
|
'msg' => 'server is not found'
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
$data = get_request_content();
|
|
|
|
$data = json_decode($data, true);
|
|
|
|
Cache::put(CacheKey::get('SERVER_VMESS_ONLINE_USER', $server->id), count($data), 3600);
|
|
|
|
Cache::put(CacheKey::get('SERVER_VMESS_LAST_PUSH_AT', $server->id), time(), 3600);
|
|
|
|
$userService = new UserService();
|
|
|
|
$formatData = [];
|
|
|
|
|
|
|
|
foreach ($data as $item) {
|
|
|
|
$formatData[$item['user_id']] = [$item['u'], $item['d']];
|
|
|
|
}
|
|
|
|
$userService->trafficFetch($server->toArray(), 'vmess', $formatData);
|
|
|
|
|
|
|
|
return response([
|
|
|
|
'ret' => 1,
|
|
|
|
'msg' => 'ok'
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// 后端获取配置
|
|
|
|
public function config(Request $request)
|
|
|
|
{
|
2023-12-06 15:01:32 -05:00
|
|
|
$request->validate([
|
|
|
|
'node_id' => 'required',
|
|
|
|
'local_port' => 'required'
|
|
|
|
],[
|
|
|
|
'node_id.required' => '节点ID不能为空',
|
|
|
|
'local_port.required' => '本地端口不能为空'
|
|
|
|
]);
|
2023-11-17 01:44:01 -05:00
|
|
|
try {
|
2023-12-06 15:01:32 -05:00
|
|
|
$json = $this->getV2RayConfig($request->input('node_id'), $request->input('local_port'));
|
2023-11-17 01:44:01 -05:00
|
|
|
} catch (\Exception $e) {
|
2023-12-06 15:01:32 -05:00
|
|
|
\Log::error($e);
|
|
|
|
throw new ApiException($e->getMessage());
|
2023-11-17 01:44:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return(json_encode($json, JSON_UNESCAPED_UNICODE));
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getV2RayConfig(int $nodeId, int $localPort)
|
|
|
|
{
|
|
|
|
$server = ServerVmess::find($nodeId);
|
|
|
|
if (!$server) {
|
2023-12-06 15:01:32 -05:00
|
|
|
return $this->fail([400,'节点不存在']);
|
2023-11-17 01:44:01 -05:00
|
|
|
}
|
|
|
|
$json = json_decode(self::V2RAY_CONFIG);
|
|
|
|
$json->log->loglevel = (int)admin_setting('server_log_enable') ? 'debug' : 'none';
|
|
|
|
$json->inbounds[1]->port = (int)$localPort;
|
|
|
|
$json->inbounds[0]->port = (int)$server->server_port;
|
|
|
|
$json->inbounds[0]->streamSettings->network = $server->network;
|
|
|
|
$this->setDns($server, $json);
|
|
|
|
$this->setNetwork($server, $json);
|
|
|
|
$this->setRule($server, $json);
|
|
|
|
$this->setTls($server, $json);
|
|
|
|
|
|
|
|
return $json;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function setDns(ServerVmess $server, object $json)
|
|
|
|
{
|
|
|
|
if ($server->dnsSettings) {
|
|
|
|
$dns = $server->dnsSettings;
|
|
|
|
if (isset($dns->servers)) {
|
|
|
|
array_push($dns->servers, '1.1.1.1');
|
|
|
|
array_push($dns->servers, 'localhost');
|
|
|
|
}
|
|
|
|
$json->dns = $dns;
|
|
|
|
$json->outbounds[0]->settings->domainStrategy = 'UseIP';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function setNetwork(ServerVmess $server, object $json)
|
|
|
|
{
|
|
|
|
if ($server->networkSettings) {
|
|
|
|
switch ($server->network) {
|
|
|
|
case 'tcp':
|
|
|
|
$json->inbounds[0]->streamSettings->tcpSettings = $server->networkSettings;
|
|
|
|
break;
|
|
|
|
case 'kcp':
|
|
|
|
$json->inbounds[0]->streamSettings->kcpSettings = $server->networkSettings;
|
|
|
|
break;
|
|
|
|
case 'ws':
|
|
|
|
$json->inbounds[0]->streamSettings->wsSettings = $server->networkSettings;
|
|
|
|
break;
|
|
|
|
case 'http':
|
|
|
|
$json->inbounds[0]->streamSettings->httpSettings = $server->networkSettings;
|
|
|
|
break;
|
|
|
|
case 'domainsocket':
|
|
|
|
$json->inbounds[0]->streamSettings->dsSettings = $server->networkSettings;
|
|
|
|
break;
|
|
|
|
case 'quic':
|
|
|
|
$json->inbounds[0]->streamSettings->quicSettings = $server->networkSettings;
|
|
|
|
break;
|
|
|
|
case 'grpc':
|
|
|
|
$json->inbounds[0]->streamSettings->grpcSettings = $server->networkSettings;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function setRule(ServerVmess $server, object $json)
|
|
|
|
{
|
|
|
|
$domainRules = array_filter(explode(PHP_EOL, admin_setting('server_v2ray_domain')));
|
|
|
|
$protocolRules = array_filter(explode(PHP_EOL, admin_setting('server_v2ray_protocol')));
|
|
|
|
if ($server->ruleSettings) {
|
|
|
|
$ruleSettings = $server->ruleSettings;
|
|
|
|
// domain
|
|
|
|
if (isset($ruleSettings->domain)) {
|
|
|
|
$ruleSettings->domain = array_filter($ruleSettings->domain);
|
|
|
|
if (!empty($ruleSettings->domain)) {
|
|
|
|
$domainRules = array_merge($domainRules, $ruleSettings->domain);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// protocol
|
|
|
|
if (isset($ruleSettings->protocol)) {
|
|
|
|
$ruleSettings->protocol = array_filter($ruleSettings->protocol);
|
|
|
|
if (!empty($ruleSettings->protocol)) {
|
|
|
|
$protocolRules = array_merge($protocolRules, $ruleSettings->protocol);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!empty($domainRules)) {
|
|
|
|
$domainObj = new \StdClass();
|
|
|
|
$domainObj->type = 'field';
|
|
|
|
$domainObj->domain = $domainRules;
|
|
|
|
$domainObj->outboundTag = 'block';
|
|
|
|
array_push($json->routing->rules, $domainObj);
|
|
|
|
}
|
|
|
|
if (!empty($protocolRules)) {
|
|
|
|
$protocolObj = new \StdClass();
|
|
|
|
$protocolObj->type = 'field';
|
|
|
|
$protocolObj->protocol = $protocolRules;
|
|
|
|
$protocolObj->outboundTag = 'block';
|
|
|
|
array_push($json->routing->rules, $protocolObj);
|
|
|
|
}
|
|
|
|
if (empty($domainRules) && empty($protocolRules)) {
|
|
|
|
$json->inbounds[0]->sniffing->enabled = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function setTls(ServerVMess $server, object $json)
|
|
|
|
{
|
|
|
|
if ((int)$server->tls) {
|
|
|
|
$tlsSettings = $server->tlsSettings;
|
|
|
|
$json->inbounds[0]->streamSettings->security = 'tls';
|
|
|
|
$tls = (object)[
|
|
|
|
'certificateFile' => '/root/.cert/server.crt',
|
|
|
|
'keyFile' => '/root/.cert/server.key'
|
|
|
|
];
|
|
|
|
$json->inbounds[0]->streamSettings->tlsSettings = new \StdClass();
|
|
|
|
if (isset($tlsSettings->serverName)) {
|
|
|
|
$json->inbounds[0]->streamSettings->tlsSettings->serverName = (string)$tlsSettings->serverName;
|
|
|
|
}
|
|
|
|
if (isset($tlsSettings->allowInsecure)) {
|
|
|
|
$json->inbounds[0]->streamSettings->tlsSettings->allowInsecure = (int)$tlsSettings->allowInsecure ? true : false;
|
|
|
|
}
|
|
|
|
$json->inbounds[0]->streamSettings->tlsSettings->certificates[0] = $tls;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|