Tool.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  1. <?php
  2. declare(strict_types=1);
  3. namespace App\Command;
  4. use App\Models\Config;
  5. use App\Models\Link;
  6. use App\Models\Node;
  7. use App\Models\User as ModelsUser;
  8. use App\Services\MFA;
  9. use App\Utils\Hash;
  10. use App\Utils\Tools;
  11. use Exception;
  12. use Ramsey\Uuid\Uuid;
  13. use Telegram\Bot\Api;
  14. use Telegram\Bot\Exceptions\TelegramSDKException;
  15. use function count;
  16. use function date;
  17. use function fgets;
  18. use function file_get_contents;
  19. use function file_put_contents;
  20. use function fwrite;
  21. use function in_array;
  22. use function json_decode;
  23. use function json_encode;
  24. use function method_exists;
  25. use function strtolower;
  26. use function trim;
  27. use const JSON_PRETTY_PRINT;
  28. use const JSON_UNESCAPED_UNICODE;
  29. use const PHP_EOL;
  30. use const STDIN;
  31. use const STDOUT;
  32. final class Tool extends Command
  33. {
  34. public string $description = <<<EOL
  35. ├─=: php xcat Tool [选项]
  36. │ ├─ setTelegram - 设置 Telegram 机器人
  37. │ ├─ resetSetting - 使用默认值覆盖数据库配置
  38. │ ├─ exportSetting - 导出数据库配置
  39. │ ├─ importSetting - 导入数据库配置
  40. │ ├─ resetNodePassword - 重置所有节点通讯密钥
  41. │ ├─ resetNodeBandwidth - 重置所有节点流量
  42. │ ├─ resetPort - 重置所有用户端口
  43. │ ├─ resetBandwidth - 重置所有用户流量
  44. │ ├─ resetPassword - 重置所有用户密码
  45. │ ├─ clearSubToken - 清除用户 Sub Token
  46. │ ├─ generateUUID - 为所有用户生成新的 UUID
  47. │ ├─ generateGa - 为所有用户生成新的 Ga Secret
  48. │ ├─ generateApiToken - 为所有用户生成新的 API Token
  49. │ ├─ setTheme - 为所有用户设置新的主题
  50. │ ├─ createAdmin - 创建管理员帐号
  51. EOL;
  52. public function boot(): void
  53. {
  54. if (count($this->argv) === 2) {
  55. echo $this->description;
  56. } else {
  57. $methodName = $this->argv[2];
  58. if (method_exists($this, $methodName)) {
  59. $this->$methodName();
  60. } else {
  61. echo '方法不存在' . PHP_EOL;
  62. }
  63. }
  64. }
  65. /**
  66. * @throws TelegramSDKException
  67. */
  68. public function setTelegram(): void
  69. {
  70. $WebhookUrl = $_ENV['baseUrl'] . '/callback/telegram?token=' . Config::obtain('telegram_request_token');
  71. $telegram = new Api(Config::obtain('telegram_token'));
  72. $telegram->removeWebhook();
  73. if ($telegram->setWebhook(['url' => $WebhookUrl])) {
  74. echo 'Bot @' . $telegram->getMe()->getUsername() . ' 设置成功!' . PHP_EOL;
  75. } else {
  76. echo '设置失败!' . PHP_EOL;
  77. }
  78. }
  79. public function resetSetting(): void
  80. {
  81. $settings = Config::all();
  82. foreach ($settings as $setting) {
  83. $setting->value = $setting->default;
  84. $setting->save();
  85. }
  86. echo '已使用默认值覆盖所有数据库设置' . PHP_EOL;
  87. }
  88. public function exportSetting(): void
  89. {
  90. $settings = Config::all();
  91. foreach ($settings as $setting) {
  92. // 因为主键自增所以即便设置为 null 也会在导入时自动分配 id
  93. // 同时避免多位开发者 pull request 时 settings.json 文件 id 重复所可能导致的冲突
  94. $setting->id = null;
  95. // 避免开发者调试配置泄露
  96. $setting->value = $setting->default;
  97. }
  98. $json_settings = json_encode($settings, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE);
  99. file_put_contents('./config/settings.json', $json_settings);
  100. echo '已导出所有数据库设置' . PHP_EOL;
  101. }
  102. public function importSetting(): void
  103. {
  104. $json_settings = file_get_contents('./config/settings.json');
  105. $settings = json_decode($json_settings, true);
  106. $config = [];
  107. $add_counter = 0;
  108. $update_counter = 0;
  109. $del_counter = 0;
  110. // 检查新增
  111. foreach ($settings as $item) {
  112. $config[] = $item['item'];
  113. $item_name = $item['item'];
  114. $query = (new Config())->where('item', $item['item'])->first();
  115. if ($query === null) {
  116. $new_item = new Config();
  117. $new_item->id = null;
  118. $new_item->item = $item['item'];
  119. $new_item->value = $item['value'];
  120. $new_item->class = $item['class'];
  121. $new_item->is_public = $item['is_public'];
  122. $new_item->type = $item['type'];
  123. $new_item->default = $item['default'];
  124. $new_item->mark = $item['mark'];
  125. $new_item->save();
  126. echo '添加新数据库设置:' . $item_name . PHP_EOL;
  127. $add_counter += 1;
  128. continue;
  129. }
  130. if ($query->class !== $item['class']) {
  131. $query->class = $item['class'];
  132. $query->save();
  133. echo '更新数据库设置:' . $item_name . PHP_EOL;
  134. $update_counter += 1;
  135. }
  136. }
  137. // 检查移除
  138. $db_settings = Config::all();
  139. foreach ($db_settings as $db_setting) {
  140. if (! in_array($db_setting->item, $config)) {
  141. $db_setting->delete();
  142. $del_counter += 1;
  143. }
  144. }
  145. if ($add_counter !== 0) {
  146. echo '添加了 ' . $add_counter . ' 项新数据库设置' . PHP_EOL;
  147. }
  148. if ($update_counter !== 0) {
  149. echo '更新了 ' . $update_counter . ' 项数据库设置' . PHP_EOL;
  150. }
  151. if ($del_counter !== 0) {
  152. echo '移除了 ' . $del_counter . ' 项数据库设置' . PHP_EOL;
  153. }
  154. }
  155. public function resetNodePassword(): void
  156. {
  157. $nodes = Node::all();
  158. foreach ($nodes as $node) {
  159. $node->password = Tools::genRandomChar(32);
  160. $node->save();
  161. }
  162. echo '已重置所有节点密码' . PHP_EOL;
  163. }
  164. public function resetNodeBandwidth(): void
  165. {
  166. $nodes = Node::all();
  167. foreach ($nodes as $node) {
  168. $node->node_bandwidth = 0;
  169. $node->save();
  170. }
  171. echo '已重置所有节点流量' . PHP_EOL;
  172. }
  173. /**
  174. * 重置所有用户端口
  175. */
  176. public function resetPort(): void
  177. {
  178. $users = ModelsUser::all();
  179. if (count($users) === 0 || count($users) >= 65535) {
  180. echo '无效的用户数量' . PHP_EOL;
  181. return;
  182. }
  183. (new ModelsUser())->update([
  184. 'port' => 0,
  185. ]);
  186. foreach ($users as $user) {
  187. $user->port = Tools::getSsPort();
  188. $user->save();
  189. }
  190. echo '已重置所有用户端口' . PHP_EOL;
  191. }
  192. /**
  193. * 重置所有用户流量
  194. */
  195. public function resetBandwidth(): void
  196. {
  197. (new ModelsUser())->where('is_banned', 0)->update([
  198. 'd' => 0,
  199. 'u' => 0,
  200. 'transfer_today' => 0,
  201. ]);
  202. echo '已重置所有用户流量' . PHP_EOL;
  203. }
  204. /**
  205. * 重置所有用户密码
  206. */
  207. public function resetPassword(): void
  208. {
  209. $users = ModelsUser::all();
  210. foreach ($users as $user) {
  211. $user->pass = Hash::passwordHash(Tools::genRandomChar(32));
  212. $user->save();
  213. }
  214. echo '已重置所有用户密码' . PHP_EOL;
  215. }
  216. /**
  217. * 清除用户 Sub Token
  218. */
  219. public function clearSubToken(): void
  220. {
  221. Link::query()->truncate();
  222. echo '已清除所有用户 Sub Token' . PHP_EOL;
  223. }
  224. /**
  225. * 为所有用户生成新的 UUID
  226. */
  227. public function generateUUID(): void
  228. {
  229. $users = ModelsUser::all();
  230. foreach ($users as $user) {
  231. $user->uuid = Uuid::uuid4();
  232. $user->save();
  233. }
  234. echo '已为所有用户生成新的 UUID' . PHP_EOL;
  235. }
  236. /**
  237. * 二次验证
  238. */
  239. public function generateGa(): void
  240. {
  241. $users = ModelsUser::all();
  242. foreach ($users as $user) {
  243. try {
  244. $user->ga_token = MFA::generateGaToken();
  245. $user->save();
  246. } catch (Exception $e) {
  247. echo $e->getMessage();
  248. }
  249. }
  250. echo '已为所有用户生成新的 Ga Secret' . PHP_EOL;
  251. }
  252. /**
  253. * 为所有用户生成新的 Api Token
  254. */
  255. public function generateApiToken(): void
  256. {
  257. $users = ModelsUser::all();
  258. foreach ($users as $user) {
  259. $user->generateApiToken();
  260. }
  261. echo '已为所有用户生成新的 Api Token' . PHP_EOL;
  262. }
  263. /**
  264. * 为所有用户设置新的主题
  265. */
  266. public function setTheme(): void
  267. {
  268. fwrite(STDOUT, '请输入要设置的主题名称: ');
  269. $theme = trim(fgets(STDIN));
  270. $users = ModelsUser::all();
  271. foreach ($users as $user) {
  272. $user->theme = $theme;
  273. $user->save();
  274. }
  275. echo '已为所有用户设置新的主题: ' . $theme . PHP_EOL;
  276. }
  277. /**
  278. * 创建 Admin 账户
  279. *
  280. * @throws Exception
  281. */
  282. public function createAdmin(): void
  283. {
  284. $y = '';
  285. $email = '';
  286. $passwd = '';
  287. if (count($this->argv) === 3) {
  288. // ask for input
  289. echo '(1/3) 请输入管理员邮箱:' . PHP_EOL;
  290. // get input
  291. $email = trim(fgets(STDIN));
  292. // write input back
  293. echo '(2/3) 请输入管理员账户密码:' . PHP_EOL;
  294. $passwd = trim(fgets(STDIN));
  295. echo '(3/3) 按 Y 或 y 确认创建:';
  296. $y = trim(fgets(STDIN));
  297. } elseif (count($this->argv) === 5) {
  298. [,,, $email, $passwd] = $this->argv;
  299. $y = 'y';
  300. }
  301. if (strtolower($y) === 'y') {
  302. // do reg user
  303. $user = new ModelsUser();
  304. $user->user_name = 'Admin';
  305. $user->email = $email;
  306. $user->remark = '';
  307. $user->pass = Hash::passwordHash($passwd);
  308. $user->passwd = Tools::genRandomChar(16);
  309. $user->uuid = Uuid::uuid4();
  310. $user->api_token = Uuid::uuid4();
  311. $user->port = Tools::getSsPort();
  312. $user->u = 0;
  313. $user->d = 0;
  314. $user->transfer_enable = 0;
  315. $user->ref_by = 0;
  316. $user->is_admin = 1;
  317. $user->reg_date = date('Y-m-d H:i:s');
  318. $user->money = 0;
  319. $user->im_type = 0;
  320. $user->im_value = '';
  321. $user->class = 0;
  322. $user->node_iplimit = 0;
  323. $user->node_speedlimit = 0;
  324. $user->theme = $_ENV['theme'];
  325. $user->locale = $_ENV['locale'];
  326. $user->ga_token = MFA::generateGaToken();
  327. $user->ga_enable = 0;
  328. if ($user->save()) {
  329. echo '创建成功,请在主页登录' . PHP_EOL;
  330. } else {
  331. echo '创建失败,请检查数据库配置' . PHP_EOL;
  332. }
  333. } else {
  334. echo '已取消创建' . PHP_EOL;
  335. }
  336. }
  337. }