PoseidonController.php 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. <?php
  2. namespace App\Http\Controllers\Server;
  3. use App\Services\ServerService;
  4. use App\Services\UserService;
  5. use App\Utils\CacheKey;
  6. use Illuminate\Http\Request;
  7. use App\Http\Controllers\Controller;
  8. use App\Models\User;
  9. use App\Models\Plan;
  10. use App\Models\Server;
  11. use App\Models\ServerLog;
  12. use Illuminate\Support\Facades\Log;
  13. use Illuminate\Support\Facades\Cache;
  14. class PoseidonController extends Controller
  15. {
  16. CONST SERVER_CONFIG = '{"api":{"services":["HandlerService","StatsService"],"tag":"api"},"stats":{},"inbound":{"port":443,"protocol":"vmess","settings":{"clients":[]},"sniffing":{"enabled": true,"destOverride": ["http","tls"]},"streamSettings":{"network":"tcp"},"tag":"proxy"},"inboundDetour":[{"listen":"0.0.0.0","port":23333,"protocol":"dokodemo-door","settings":{"address":"0.0.0.0"},"tag":"api"}],"log":{"loglevel":"debug","access":"access.log","error":"error.log"},"outbound":{"protocol":"freedom","settings":{}},"outboundDetour":[{"protocol":"blackhole","settings":{},"tag":"block"}],"routing":{"rules":[{"inboundTag":"api","outboundTag":"api","type":"field"}]},"policy":{"levels":{"0":{"handshake":4,"connIdle":300,"uplinkOnly":5,"downlinkOnly":30,"statsUserUplink":true,"statsUserDownlink":true}}}}';
  17. public $poseidonVersion;
  18. public function __construct(Request $request)
  19. {
  20. $this->poseidonVersion = $request->input('poseidon_version');
  21. }
  22. // 后端获取用户
  23. public function user(Request $request)
  24. {
  25. if ($r = $this->verifyToken($request)) { return $r; }
  26. $nodeId = $request->input('node_id');
  27. $server = Server::find($nodeId);
  28. if (!$server) {
  29. return $this->error("server could not be found", 404);
  30. }
  31. Cache::put(CacheKey::get('SERVER_LAST_CHECK_AT', $server->id), time(), 3600);
  32. $serverService = new ServerService();
  33. $users = $serverService->getAvailableUsers(json_decode($server->group_id));
  34. $result = [];
  35. foreach ($users as $user) {
  36. $user->v2ray_user = [
  37. "uuid" => $user->uuid,
  38. "email" => sprintf("%s@v2board.user", $user->uuid),
  39. "alter_id" => $user->v2ray_alter_id,
  40. "level" => $user->v2ray_level,
  41. ];
  42. unset($user['uuid']);
  43. unset($user['v2ray_alter_id']);
  44. unset($user['v2ray_level']);
  45. array_push($result, $user);
  46. }
  47. return $this->success($result);
  48. }
  49. // 后端提交数据
  50. public function submit(Request $request)
  51. {
  52. if ($r = $this->verifyToken($request)) { return $r; }
  53. $server = Server::find($request->input('node_id'));
  54. if (!$server) {
  55. return $this->error("server could not be found", 404);
  56. }
  57. $data = file_get_contents('php://input');
  58. $data = json_decode($data, true);
  59. Cache::put(CacheKey::get('SERVER_ONLINE_USER', $server->id), count($data), 3600);
  60. $serverService = new ServerService();
  61. $userService = new UserService();
  62. foreach ($data as $item) {
  63. $u = $item['u'] * $server->rate;
  64. $d = $item['d'] * $server->rate;
  65. if (!$userService->trafficFetch($u, $d, $item['user_id'])) {
  66. return $this->error("user fetch fail", 500);
  67. }
  68. $serverService->log(
  69. $item['user_id'],
  70. $request->input('node_id'),
  71. $item['u'],
  72. $item['d'],
  73. $server->rate,
  74. 'vmess'
  75. );
  76. }
  77. return $this->success('');
  78. }
  79. // 后端获取配置
  80. public function config(Request $request)
  81. {
  82. if ($r = $this->verifyToken($request)) { return $r; }
  83. $nodeId = $request->input('node_id');
  84. $localPort = $request->input('local_port');
  85. if (empty($nodeId) || empty($localPort)) {
  86. return $this->error('invalid parameters', 400);
  87. }
  88. $serverService = new ServerService();
  89. try {
  90. $json = $serverService->getConfig($nodeId, $localPort);
  91. $json->poseidon = [
  92. 'license_key' => (string)config('v2board.server_license'),
  93. ];
  94. if ($this->poseidonVersion >= 'v1.5.0') {
  95. // don't need it after v1.5.0
  96. unset($json->inboundDetour);
  97. unset($json->stats);
  98. unset($json->api);
  99. array_shift($json->routing->rules);
  100. }
  101. foreach($json->policy->levels as &$level) {
  102. $level->handshake = 2;
  103. $level->uplinkOnly = 2;
  104. $level->downlinkOnly = 2;
  105. $level->connIdle = 60;
  106. }
  107. return $this->success($json);
  108. } catch (\Exception $e) {
  109. return $this->error($e->getMessage(), 500);
  110. }
  111. }
  112. protected function verifyToken(Request $request)
  113. {
  114. $token = $request->input('token');
  115. if (empty($token)) {
  116. return $this->error("token must be set");
  117. }
  118. if ($token !== config('v2board.server_token')) {
  119. return $this->error("invalid token");
  120. }
  121. }
  122. protected function error($msg, int $status = 400) {
  123. return response([
  124. 'msg' => $msg,
  125. ], $status);
  126. }
  127. protected function success($data) {
  128. return response([
  129. 'msg' => 'ok',
  130. 'data' => $data,
  131. ]);
  132. }
  133. }