PoseidonController.php 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. <?php
  2. namespace App\Http\Controllers\Server;
  3. use Illuminate\Http\Request;
  4. use App\Http\Controllers\Controller;
  5. use App\Models\User;
  6. use App\Models\Plan;
  7. use App\Models\Server;
  8. use App\Models\ServerLog;
  9. use Illuminate\Support\Facades\Log;
  10. use Illuminate\Support\Facades\Cache;
  11. class PoseidonController extends Controller
  12. {
  13. 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}}}}';
  14. // 后端获取用户
  15. public function user(Request $request)
  16. {
  17. if ($r = $this->verifyToken($request)) { return $r; }
  18. $nodeId = $request->input('node_id');
  19. $server = Server::find($nodeId);
  20. if (!$server) {
  21. return $this->error("server could not be found", 404);
  22. }
  23. Cache::put('server_last_check_at_' . $server->id, time());
  24. $users = User::whereIn('group_id', json_decode($server->group_id))
  25. ->select([
  26. 'id',
  27. 'email',
  28. 't',
  29. 'u',
  30. 'd',
  31. 'transfer_enable',
  32. 'enable',
  33. 'v2ray_uuid',
  34. 'v2ray_alter_id',
  35. 'v2ray_level'
  36. ])
  37. ->whereRaw('u + d < transfer_enable')
  38. ->where('enable', 1)
  39. ->get();
  40. $result = [];
  41. foreach ($users as $user) {
  42. $user->v2ray_user = [
  43. "uuid" => $user->v2ray_uuid,
  44. "email" => sprintf("%s@v2board.user", $user->v2ray_uuid),
  45. "alter_id" => $user->v2ray_alter_id,
  46. "level" => $user->v2ray_level,
  47. ];
  48. unset($user['v2ray_uuid']);
  49. unset($user['v2ray_alter_id']);
  50. unset($user['v2ray_level']);
  51. array_push($result, $user);
  52. }
  53. return $this->success($result);
  54. }
  55. // 后端提交数据
  56. public function submit(Request $request)
  57. {
  58. if ($r = $this->verifyToken($request)) { return $r; }
  59. Log::info('serverSubmitData:' . $request->input('node_id') . ':' . file_get_contents('php://input'));
  60. $server = Server::find($request->input('node_id'));
  61. if (!$server) {
  62. return $this->error("server could not be found", 404);
  63. }
  64. $data = file_get_contents('php://input');
  65. $data = json_decode($data, true);
  66. foreach ($data as $item) {
  67. $u = $item['u'] * $server->rate;
  68. $d = $item['d'] * $server->rate;
  69. $user = User::find($item['user_id']);
  70. $user->t = time();
  71. $user->u = $user->u + $u;
  72. $user->d = $user->d + $d;
  73. $user->save();
  74. $serverLog = new ServerLog();
  75. $serverLog->user_id = $item['user_id'];
  76. $serverLog->server_id = $request->input('node_id');
  77. $serverLog->u = $item['u'];
  78. $serverLog->d = $item['d'];
  79. $serverLog->rate = $server->rate;
  80. $serverLog->save();
  81. }
  82. return $this->success('');
  83. }
  84. // 后端获取配置
  85. public function config(Request $request)
  86. {
  87. if ($r = $this->verifyToken($request)) { return $r; }
  88. $nodeId = $request->input('node_id');
  89. $localPort = $request->input('local_port');
  90. if (empty($nodeId) || empty($localPort)) {
  91. return $this->error('invalid parameters', 400);
  92. }
  93. $server = Server::find($nodeId);
  94. if (!$server) {
  95. return $this->error("server could not be found", 404);
  96. }
  97. $json = json_decode(self::SERVER_CONFIG);
  98. $json->inboundDetour[0]->port = (int)$localPort;
  99. $json->inbound->port = (int)$server->server_port;
  100. $json->inbound->streamSettings->network = $server->network;
  101. if ($server->settings) {
  102. switch ($server->network) {
  103. case 'tcp':
  104. $json->inbound->streamSettings->tcpSettings = json_decode($server->settings);
  105. break;
  106. case 'kcp':
  107. $json->inbound->streamSettings->kcpSettings = json_decode($server->settings);
  108. break;
  109. case 'ws':
  110. $json->inbound->streamSettings->wsSettings = json_decode($server->settings);
  111. break;
  112. case 'http':
  113. $json->inbound->streamSettings->httpSettings = json_decode($server->settings);
  114. break;
  115. case 'domainsocket':
  116. $json->inbound->streamSettings->dsSettings = json_decode($server->settings);
  117. break;
  118. case 'quic':
  119. $json->inbound->streamSettings->quicSettings = json_decode($server->settings);
  120. break;
  121. }
  122. }
  123. if ($server->rules) {
  124. $rules = json_decode($server->rules);
  125. // domain
  126. if (isset($rules->domain)) {
  127. $domainObj = new \StdClass();
  128. $domainObj->type = 'field';
  129. $domainObj->domain = $rules->domain;
  130. $domainObj->outboundTag = 'block';
  131. array_push($json->routing->rules, $domainObj);
  132. }
  133. // protocol
  134. if (isset($rules->protocol)) {
  135. $protocolObj = new \StdClass();
  136. $protocolObj->type = 'field';
  137. $protocolObj->protocol = $rules->protocol;
  138. $protocolObj->outboundTag = 'block';
  139. array_push($json->routing->rules, $protocolObj);
  140. }
  141. }
  142. if ((int)$server->tls) {
  143. $json->inbound->streamSettings->security = 'tls';
  144. $tls = (object)[
  145. 'certificateFile' => '/home/v2ray.crt',
  146. 'keyFile' => '/home/v2ray.key'
  147. ];
  148. $json->inbound->streamSettings->tlsSettings = new \StdClass();
  149. $json->inbound->streamSettings->tlsSettings->certificates[0] = $tls;
  150. }
  151. $json->poseidon = [
  152. 'license_key' => (string)config('v2board.server_license'),
  153. ];
  154. return $this->success($json);
  155. }
  156. protected function verifyToken(Request $request)
  157. {
  158. $token = $request->input('token');
  159. if (empty($token)) {
  160. return $this->error("token must be set");
  161. }
  162. if ($token !== config('v2board.server_token')) {
  163. return $this->error("invalid token");
  164. }
  165. }
  166. protected function error($msg, int $status = 400) {
  167. return response([
  168. 'msg' => $msg,
  169. ], $status);
  170. }
  171. protected function success($data) {
  172. return response([
  173. 'msg' => 'ok',
  174. 'data' => $data,
  175. ]);
  176. }
  177. }