Compare commits

...

474 Commits

Author SHA1 Message Date
as2252258 b14b18040b eee 2024-06-21 14:35:43 +08:00
as2252258 def6675c81 eee 2024-06-20 17:34:04 +08:00
as2252258 6cdb51dd18 eee 2024-06-20 17:30:49 +08:00
as2252258 e9367ec735 eee 2024-06-20 17:17:50 +08:00
as2252258 684c5a3ebb eee 2024-06-20 16:41:45 +08:00
as2252258 858b9bc9f9 eee 2024-04-24 14:31:06 +08:00
as2252258 4cbd1fb500 eee 2024-04-24 14:17:09 +08:00
as2252258 361acc09f2 eee 2023-12-19 17:50:58 +08:00
as2252258 bf4146c73b eee 2023-12-19 14:53:07 +08:00
as2252258 d3312c2484 eee 2023-12-18 03:31:55 +08:00
as2252258 375885fdd6 eee 2023-12-18 03:30:11 +08:00
as2252258 71431c6ee2 eee 2023-12-18 03:19:11 +08:00
as2252258 2009ba0055 eee 2023-12-18 03:18:26 +08:00
as2252258 2603463052 eee 2023-12-18 03:12:08 +08:00
as2252258 9f3978dda6 eee 2023-12-12 18:03:07 +08:00
as2252258 92e65c4abd eee 2023-12-12 15:35:34 +08:00
as2252258 4ebb7e9686 eee 2023-12-12 10:56:42 +08:00
as2252258 3d89612a8e eee 2023-12-06 17:41:45 +08:00
as2252258 be035ed69f eee 2023-12-03 01:26:55 +08:00
as2252258 bbfd7367d0 eee 2023-12-02 17:29:24 +08:00
as2252258 3dd4f8d4d4 eee 2023-12-02 17:27:22 +08:00
as2252258 2a78115643 eee 2023-11-30 17:02:19 +08:00
as2252258 04b1aab406 eee 2023-11-30 11:40:04 +08:00
as2252258 8781b4dc45 eee 2023-11-29 14:50:56 +08:00
as2252258 a0d0b6a597 eee 2023-11-29 14:49:18 +08:00
as2252258 bf930a375a eee 2023-11-29 14:37:21 +08:00
as2252258 ddfbf85fee eee 2023-11-29 14:36:28 +08:00
as2252258 6fc21e0557 eee 2023-11-29 14:33:25 +08:00
as2252258 cc715e58aa eee 2023-11-29 14:31:52 +08:00
as2252258 7189999a71 eee 2023-11-29 14:21:30 +08:00
as2252258 3bb2907407 eee 2023-11-24 10:22:14 +08:00
as2252258 2c9e0886b5 eee 2023-11-23 16:13:47 +08:00
as2252258 882062a403 eee 2023-11-22 17:07:11 +08:00
as2252258 01f920c8d8 eee 2023-11-22 17:05:25 +08:00
as2252258 d74cb4a7e3 eee 2023-11-22 10:35:49 +08:00
as2252258 09c3890ccd eee 2023-11-22 10:09:27 +08:00
as2252258 1f592689ef eee 2023-11-22 09:26:18 +08:00
as2252258 3e86f8efaa eee 2023-11-17 00:02:41 +08:00
as2252258 1540f26856 eee 2023-11-13 22:37:48 +08:00
as2252258 b5deafc076 eee 2023-11-04 00:57:06 +08:00
as2252258 f8b557d88a eee 2023-10-24 17:22:28 +08:00
as2252258 a1d80c3315 eee 2023-10-24 15:22:15 +08:00
as2252258 730aa3ec9e eee 2023-10-24 15:21:41 +08:00
as2252258 0c99ac1383 eee 2023-10-24 15:20:55 +08:00
as2252258 19d25a0be5 eee 2023-10-24 15:19:46 +08:00
as2252258 f621513587 eee 2023-10-18 10:58:24 +08:00
as2252258 8c4ba1ae99 eee 2023-10-18 10:17:09 +08:00
as2252258 2cf557b445 eee 2023-10-18 10:14:47 +08:00
as2252258 5024ca2504 eee 2023-10-17 20:32:26 +08:00
as2252258 879e347e3b eee 2023-10-17 20:13:29 +08:00
as2252258 65d1585722 eee 2023-10-17 17:18:55 +08:00
as2252258 f4752062bd eee 2023-10-17 14:50:45 +08:00
as2252258 c3ab610d83 eee 2023-08-29 21:06:05 +08:00
as2252258 0154d12f44 eee 2023-08-29 20:58:09 +08:00
as2252258 ad68daaac7 qqq 2023-08-18 21:38:57 +08:00
as2252258 70600c83da qqq 2023-08-18 21:37:50 +08:00
as2252258 436e78aea6 qqq 2023-08-18 14:32:51 +08:00
as2252258 5150b9027e qqq 2023-08-17 17:04:55 +08:00
as2252258 b26d5074fc qqq 2023-08-17 17:03:04 +08:00
as2252258 8257e675ba qqq 2023-08-17 16:56:51 +08:00
as2252258 8fe0453fe4 qqq 2023-08-17 16:05:17 +08:00
as2252258 11d1c0f708 qqq 2023-08-17 16:02:55 +08:00
as2252258 c43fc37e57 qqq 2023-08-17 15:55:05 +08:00
as2252258 bbcfcc3878 qqq 2023-08-16 17:40:01 +08:00
as2252258 27b73b26f8 qqq 2023-08-16 10:26:48 +08:00
as2252258 e42fb7f989 qqq 2023-08-16 01:01:47 +08:00
as2252258 5c91274d7b qqq 2023-08-14 22:14:35 +08:00
as2252258 86b7048d54 qqq 2023-08-14 21:25:37 +08:00
as2252258 53c06f8fd7 qqq 2023-08-14 21:09:44 +08:00
as2252258 8cce9e5a3e qqq 2023-08-12 00:02:15 +08:00
as2252258 8391388f1f qqq 2023-08-11 18:33:33 +08:00
as2252258 815785b667 qqq 2023-08-11 18:28:55 +08:00
as2252258 5d4246ef96 qqq 2023-08-11 18:26:37 +08:00
as2252258 70539e6b20 qqq 2023-08-11 18:23:24 +08:00
as2252258 72ee9b473d qqq 2023-08-11 09:40:06 +08:00
as2252258 bb4df7307a qqq 2023-08-11 02:35:46 +08:00
as2252258 3196a7cffb qqq 2023-08-11 02:32:04 +08:00
as2252258 db7e16c35c qqq 2023-08-11 02:24:22 +08:00
as2252258 7424554529 qqq 2023-08-11 02:22:29 +08:00
as2252258 65fa64e6b7 qqq 2023-08-11 02:18:04 +08:00
as2252258 8fb3fd699e qqq 2023-08-11 00:12:32 +08:00
as2252258 59419edbdf qqq 2023-07-31 23:16:59 +08:00
as2252258 98bda3d45e qqq 2023-07-31 23:13:57 +08:00
as2252258 371a1590bb qqq 2023-07-31 23:13:44 +08:00
as2252258 ba3df62cb5 qqq 2023-07-31 23:08:58 +08:00
as2252258 2b2a779b94 qqq 2023-07-26 17:49:41 +08:00
as2252258 1dedaa52cc qqq 2023-07-26 17:48:34 +08:00
as2252258 4a667a7596 qqq 2023-07-26 17:45:07 +08:00
as2252258 6d04611bb5 qqq 2023-07-26 17:43:19 +08:00
as2252258 a006f988f2 qqq 2023-07-26 17:40:31 +08:00
as2252258 61be1e5bdc qqq 2023-07-26 10:05:58 +08:00
as2252258 4326e90f4c qqq 2023-07-26 10:04:26 +08:00
as2252258 970f630bba qqq 2023-07-26 09:59:11 +08:00
as2252258 07e651d2af qqq 2023-07-26 09:58:34 +08:00
as2252258 1fe416c4c4 qqq 2023-07-20 15:02:06 +08:00
as2252258 4ada7bbd1c qqq 2023-07-20 15:01:23 +08:00
as2252258 8a72e90257 qqq 2023-07-15 01:56:39 +08:00
as2252258 8bacbb74d9 qqq 2023-07-15 01:51:38 +08:00
as2252258 d587d41851 qqq 2023-07-10 10:36:45 +08:00
as2252258 1d4073df70 qqq 2023-07-10 10:34:41 +08:00
as2252258 b7e347162e qqq 2023-07-10 10:29:39 +08:00
as2252258 00cbeb819c qqq 2023-07-10 10:24:01 +08:00
as2252258 611ae77bed qqq 2023-07-10 02:22:38 +08:00
as2252258 63a89745d9 qqq 2023-07-10 02:20:20 +08:00
as2252258 d09cd983d7 qqq 2023-07-06 17:57:10 +08:00
as2252258 c7dd16b4c9 qqq 2023-07-06 16:53:53 +08:00
as2252258 d1cf63591d qqq 2023-07-06 16:22:19 +08:00
as2252258 2a65994a85 qqq 2023-07-06 16:20:46 +08:00
as2252258 f04f59dd6f qqq 2023-07-06 16:17:00 +08:00
as2252258 6aa764fb86 qqq 2023-07-06 16:00:02 +08:00
as2252258 136b0dc1a6 qqq 2023-07-06 15:38:03 +08:00
as2252258 f8dd9efbb7 qqq 2023-06-28 15:21:01 +08:00
as2252258 c9aac1f196 qqq 2023-06-27 16:45:00 +08:00
as2252258 ffd877dbdf qqq 2023-06-27 16:29:09 +08:00
as2252258 e77d52f68d qqq 2023-06-12 17:21:48 +08:00
as2252258 6583352a5e qqq 2023-06-12 17:20:42 +08:00
as2252258 ebcb8acfce qqq 2023-06-12 17:06:20 +08:00
as2252258 14507e8b01 qqq 2023-06-12 16:36:03 +08:00
as2252258 b5d5826fa9 qqq 2023-06-12 15:34:08 +08:00
as2252258 30e611a655 qqq 2023-06-12 15:33:47 +08:00
as2252258 8248b418ad qqq 2023-06-12 15:31:44 +08:00
as2252258 599e353a28 qqq 2023-05-29 14:32:32 +08:00
as2252258 5808493e46 qqq 2023-05-26 18:42:55 +08:00
as2252258 cd876df1c9 r 2023-05-26 18:38:15 +08:00
as2252258 57dbb1e109 qqq 2023-05-26 18:28:07 +08:00
as2252258 29357fbfb9 qqq 2023-05-26 18:26:45 +08:00
as2252258 a42daff851 qqq 2023-05-26 16:07:12 +08:00
as2252258 4b16329898 qqq 2023-05-26 16:05:21 +08:00
as2252258 5c823adc59 qqq 2023-05-26 16:02:33 +08:00
as2252258 7351d0a2f4 qqq 2023-05-26 11:25:43 +08:00
as2252258 2e36448282 qqq 2023-05-26 11:23:21 +08:00
as2252258 b027823e54 qqq 2023-05-26 11:21:36 +08:00
as2252258 5f68ae9e48 qqq 2023-05-26 11:16:43 +08:00
as2252258 9327b74bdc qqq 2023-05-26 10:43:28 +08:00
as2252258 d58f41d123 qqq 2023-05-26 10:16:20 +08:00
as2252258 76cb95a184 qqq 2023-05-25 16:59:17 +08:00
as2252258 4efac235a0 变更 2023-04-24 13:35:16 +08:00
as2252258 385c297f56 变更 2023-04-24 11:51:45 +08:00
as2252258 54f3487b30 变更 2023-04-24 11:50:58 +08:00
as2252258 714dc328a8 变更 2023-04-24 11:42:43 +08:00
as2252258 c3c4340c17 变更 2023-04-24 11:09:38 +08:00
as2252258 a9bae3be41 变更 2023-04-24 10:54:56 +08:00
as2252258 76325af578 变更 2023-04-23 18:34:13 +08:00
as2252258 b02beb658d 变更 2023-04-23 18:15:25 +08:00
as2252258 9ee260870f 变更 2023-04-23 10:37:26 +08:00
as2252258 2f6b14f5f7 变更 2023-04-22 02:47:37 +08:00
as2252258 e8c4a729a5 变更 2023-04-22 02:46:58 +08:00
as2252258 d501e85153 变更 2023-04-22 02:30:02 +08:00
as2252258 d1fc1ae874 变更 2023-04-22 02:28:26 +08:00
as2252258 487d261134 变更 2023-04-22 02:23:34 +08:00
as2252258 ace208f467 变更 2023-04-22 02:16:02 +08:00
as2252258 61bd740832 变更 2023-04-22 02:13:01 +08:00
as2252258 e3a5980648 变更 2023-04-22 02:11:07 +08:00
as2252258 ba81504942 变更 2023-04-22 02:04:31 +08:00
as2252258 3c0581c2f6 变更 2023-04-21 23:17:55 +08:00
as2252258 cc7a2d95ad 变更 2023-04-21 23:13:22 +08:00
as2252258 eb4cc76eee 变更 2023-04-21 23:09:35 +08:00
as2252258 f313860436 变更 2023-04-21 22:26:43 +08:00
as2252258 ada0f49a43 变更 2023-04-19 15:35:57 +08:00
as2252258 09cf74abdf 变更 2023-04-19 15:23:38 +08:00
as2252258 48444cd0fa 变更 2023-04-19 15:21:09 +08:00
as2252258 8ca59cc91d 变更 2023-04-19 15:12:57 +08:00
as2252258 2534cd0d1b 变更 2023-04-19 15:12:34 +08:00
as2252258 12b2b0fe48 变更 2023-04-19 15:05:28 +08:00
as2252258 28c139fa50 变更 2023-04-19 14:47:52 +08:00
as2252258 c36abf43f6 变更 2023-04-19 14:46:38 +08:00
as2252258 71caa4880a 变更 2023-04-19 14:41:32 +08:00
as2252258 6c21470dbb 变更 2023-04-19 14:37:25 +08:00
as2252258 28e7e5dc81 变更 2023-04-19 14:32:42 +08:00
as2252258 73a0b50a98 变更 2023-04-19 14:29:33 +08:00
as2252258 e6da301351 变更 2023-04-19 14:29:22 +08:00
as2252258 e0203e2ccd 变更 2023-04-19 13:38:08 +08:00
as2252258 9f424953ab 变更 2023-04-19 13:32:49 +08:00
as2252258 89e72ceea1 变更 2023-04-19 13:28:55 +08:00
as2252258 ff5097dc59 变更 2023-04-19 13:15:35 +08:00
as2252258 3b0d556700 变更 2023-04-19 13:15:21 +08:00
as2252258 c739912494 变更 2023-04-19 13:04:17 +08:00
as2252258 2e323b6a1f 变更 2023-04-18 23:59:29 +08:00
as2252258 57ff181110 变更 2023-04-18 23:48:42 +08:00
as2252258 1086534302 变更 2023-04-18 23:47:30 +08:00
as2252258 59c4da5a9c 变更 2023-04-18 22:23:41 +08:00
as2252258 324646c136 变更 2023-04-18 22:20:20 +08:00
as2252258 2c5cf3e5c8 变更 2023-04-17 01:48:27 +08:00
as2252258 642ec973cd 变更 2023-04-17 01:29:43 +08:00
as2252258 dbfb477023 变更 2023-04-17 01:07:14 +08:00
as2252258 a43f9586ec 变更 2023-04-17 01:02:14 +08:00
as2252258 cf2881ab98 变更 2023-04-17 01:01:45 +08:00
as2252258 c2308f2188 变更 2023-04-16 15:25:46 +08:00
as2252258 8c0f98da2a 变更 2023-04-16 15:24:54 +08:00
as2252258 634b4f7aa3 变更 2023-04-16 14:41:20 +08:00
as2252258 382c94f44d 变更 2023-04-16 02:48:14 +08:00
as2252258 03ee9b4275 变更 2023-04-16 02:46:54 +08:00
as2252258 d3897d3337 变更 2023-04-16 02:41:03 +08:00
as2252258 af9a14dfdc 变更 2023-04-16 02:40:09 +08:00
as2252258 2273b3d1bd 变更 2023-04-16 02:32:36 +08:00
as2252258 4e390e89d6 变更 2023-04-16 02:28:58 +08:00
as2252258 e0159c836a 变更 2023-04-16 02:26:24 +08:00
as2252258 a6d2d97fb5 变更 2023-04-16 02:15:51 +08:00
as2252258 5ec3fed09d 变更 2023-04-16 02:01:26 +08:00
as2252258 1e3f643fa9 变更 2023-04-16 00:15:09 +08:00
as2252258 666e755cd2 变更 2023-04-15 23:48:47 +08:00
as2252258 7792bc0a0b 变更 2023-04-04 14:19:48 +08:00
as2252258 1ade082657 变更 2023-04-04 14:18:36 +08:00
as2252258 f46e63d876 变更 2023-04-04 14:05:12 +08:00
as2252258 ad4e331760 变更 2023-04-04 13:59:35 +08:00
as2252258 6e43e9319a 变更 2023-04-04 13:56:42 +08:00
as2252258 6dd6dfdcf3 变更 2023-04-04 13:55:38 +08:00
as2252258 e9a0aad626 变更 2023-04-03 14:33:09 +08:00
as2252258 0f870e4268 变更 2023-04-03 11:08:10 +08:00
as2252258 22f1dec147 变更 2023-04-03 00:51:46 +08:00
as2252258 b98fed7b7c 变更 2023-04-02 23:50:29 +08:00
as2252258 b7ee0b3b62 变更 2023-04-02 23:49:09 +08:00
as2252258 5138c429fe 变更 2023-04-02 23:34:07 +08:00
as2252258 bae719f091 变更 2023-04-02 23:32:55 +08:00
as2252258 33920ece68 变更 2023-04-02 00:50:42 +08:00
as2252258 664cc98934 变更 2023-04-02 00:32:35 +08:00
as2252258 d57c803a53 变更 2023-04-01 23:57:36 +08:00
as2252258 881c1b3d49 变更 2023-04-01 23:56:30 +08:00
as2252258 5e6ffd9ae5 变更 2023-04-01 23:49:43 +08:00
as2252258 8049805435 变更 2023-04-01 23:12:58 +08:00
as2252258 6ec7d346de 变更 2023-04-01 22:38:04 +08:00
as2252258 12f70016e2 变更 2023-04-01 22:36:17 +08:00
as2252258 a2a1699ef6 变更 2023-03-31 10:29:28 +08:00
as2252258 518318ddc8 变更 2023-03-31 10:26:04 +08:00
as2252258 71eb325e80 变更 2023-03-31 10:24:29 +08:00
as2252258 5c861c312b 变更 2023-03-30 23:02:12 +08:00
as2252258 65cbde3706 变更 2023-03-30 23:01:51 +08:00
as2252258 ea90aebd62 变更 2023-03-30 18:24:05 +08:00
as2252258 f922c209a8 变更 2023-02-13 11:50:27 +08:00
as2252258 49fae2129f 变更 2023-02-07 16:55:21 +08:00
as2252258 3e201bdbdd 变更 2022-12-12 17:31:11 +08:00
as2252258 ffdd692f99 变更 2022-10-25 14:58:30 +08:00
as2252258 b74abb5500 变更 2022-10-11 23:28:51 +08:00
as2252258 d3ef6da34a 变更 2022-10-11 23:27:56 +08:00
as2252258 cc095d2866 变更 2022-10-11 18:55:06 +08:00
as2252258 cedcb89fe4 变更 2022-10-11 18:41:58 +08:00
as2252258 707bd69eda 变更 2022-10-11 15:15:04 +08:00
as2252258 0099914745 变更 2022-09-25 17:02:49 +08:00
as2252258 bb8310dcf3 变更 2022-09-25 04:19:00 +08:00
as2252258 13b9344743 变更 2022-09-23 18:55:45 +08:00
as2252258 b3a58b0528 变更 2022-09-20 18:25:55 +08:00
as2252258 fa70c1facc 变更 2022-09-07 13:54:21 +08:00
as2252258 391d27b8ea change file 2022-09-07 13:52:51 +08:00
as2252258 f1dac3ebcf 变更 2022-09-07 13:48:49 +08:00
as2252258 e7ace87826 modify plugin name 2022-07-11 17:45:15 +08:00
as2252258 0c54dfe2af modify plugin name 2022-07-11 16:34:13 +08:00
as2252258 5b46f20b9d modify plugin name 2022-07-11 16:31:57 +08:00
as2252258 4ebd409853 modify plugin name 2022-07-11 16:09:57 +08:00
as2252258 9a0294eaa2 modify plugin name 2022-06-23 13:44:31 +08:00
as2252258 ebc5cf7f8b modify plugin name 2022-06-23 11:20:57 +08:00
as2252258 43bb7c97d6 modify plugin name 2022-06-23 10:44:06 +08:00
as2252258 9d9fdba690 modify plugin name 2022-06-23 10:37:06 +08:00
as2252258 e35fca4638 变更 2022-06-23 01:20:23 +08:00
as2252258 47f1a6c4f5 变更 2022-06-23 01:18:19 +08:00
as2252258 ab6c582025 变更 2022-06-23 00:26:23 +08:00
as2252258 c880992006 modify plugin name 2022-06-22 19:16:07 +08:00
as2252258 0c4c973570 modify plugin name 2022-06-22 18:58:26 +08:00
as2252258 2301b9a479 modify plugin name 2022-06-22 18:55:51 +08:00
as2252258 329666f9b4 modify plugin name 2022-06-22 18:55:30 +08:00
as2252258 16ea9f3848 modify plugin name 2022-06-22 18:43:36 +08:00
as2252258 f83afa3e71 modify plugin name 2022-06-22 18:12:30 +08:00
as2252258 660b1a9bec modify plugin name 2022-06-22 17:42:17 +08:00
as2252258 b02213393f modify plugin name 2022-06-22 17:41:32 +08:00
as2252258 4a6aa23e17 modify plugin name 2022-06-22 17:16:17 +08:00
as2252258 62246c7f74 modify plugin name 2022-06-22 17:06:29 +08:00
as2252258 5f3d60cca9 modify plugin name 2022-06-22 16:50:37 +08:00
as2252258 b85ba031f6 modify plugin name 2022-06-22 16:46:22 +08:00
as2252258 dcc380e6b2 modify plugin name 2022-06-22 16:29:41 +08:00
as2252258 ffa0c8c7f6 modify plugin name 2022-06-22 10:53:58 +08:00
as2252258 ea7618ed1b modify plugin name 2022-06-22 10:23:09 +08:00
as2252258 8281a0fecb modify plugin name 2022-06-22 10:21:56 +08:00
as2252258 a3696c87e3 modify plugin name 2022-06-22 10:13:44 +08:00
as2252258 fc01618c98 modify plugin name 2022-06-20 18:45:03 +08:00
as2252258 43816c8c18 modify plugin name 2022-06-20 18:27:25 +08:00
as2252258 af5027cdd3 modify plugin name 2022-06-20 18:20:51 +08:00
as2252258 eca09d0c1d modify plugin name 2022-06-20 18:18:38 +08:00
as2252258 2a5b7d1cac modify plugin name 2022-06-20 18:14:25 +08:00
as2252258 b20c8fc5a3 modify plugin name 2022-06-20 18:00:00 +08:00
as2252258 5e8d9ceae7 modify plugin name 2022-06-20 17:43:00 +08:00
as2252258 df7299d588 modify plugin name 2022-06-20 17:33:27 +08:00
as2252258 0b65995ce1 modify plugin name 2022-06-20 17:33:02 +08:00
as2252258 b0b7906135 modify plugin name 2022-06-20 17:31:04 +08:00
as2252258 005b33c9d6 modify plugin name 2022-06-20 17:25:01 +08:00
as2252258 06db2fee3a modify plugin name 2022-06-17 13:55:46 +08:00
as2252258 03fe548c4e modify plugin name 2022-06-17 12:39:44 +08:00
as2252258 2f5b55ea1f modify plugin name 2022-06-17 12:33:14 +08:00
as2252258 f85d2aa0a2 modify plugin name 2022-06-17 12:27:33 +08:00
as2252258 7fdf83cd3a modify plugin name 2022-06-17 11:59:19 +08:00
as2252258 e3d9e7d5ec modify plugin name 2022-06-17 10:43:28 +08:00
as2252258 fb5b3ed27d modify plugin name 2022-06-17 09:57:57 +08:00
as2252258 12b59eaaa3 modify plugin name 2022-06-16 19:00:48 +08:00
as2252258 58b285b9e0 modify plugin name 2022-06-16 18:23:24 +08:00
as2252258 2f32a96966 modify plugin name 2022-06-16 18:17:17 +08:00
as2252258 67e7a96323 modify plugin name 2022-06-16 18:13:27 +08:00
as2252258 979808d9b8 modify plugin name 2022-06-16 18:07:27 +08:00
as2252258 d08daca870 modify plugin name 2022-06-16 18:00:11 +08:00
as2252258 314c9076a4 modify plugin name 2022-06-16 17:58:29 +08:00
as2252258 d2f2335107 modify plugin name 2022-06-16 17:51:22 +08:00
as2252258 35ce71a8cc modify plugin name 2022-06-16 17:49:06 +08:00
as2252258 0051f1f15c modify plugin name 2022-06-16 17:38:22 +08:00
as2252258 0233acb279 modify plugin name 2022-06-08 15:36:27 +08:00
as2252258 986a271e88 modify plugin name 2022-06-08 15:34:34 +08:00
as2252258 d11def7736 modify plugin name 2022-06-08 15:33:03 +08:00
as2252258 44b2cf8ee4 modify plugin name 2022-06-08 15:31:17 +08:00
as2252258 e47c7f0813 e 2022-06-08 14:57:23 +08:00
as2252258 0faa8f3ec4 modify plugin name 2022-06-08 14:53:13 +08:00
as2252258 9db7143ed6 变更 2022-05-31 11:37:00 +08:00
as2252258 273f7b3ee5 变更 2022-05-03 06:56:28 +08:00
as2252258 6b276fd1c2 modify mysql result 2022-04-29 17:08:35 +08:00
as2252258 8a1e001cab modify mysql result 2022-04-29 17:06:34 +08:00
as2252258 804f795f98 modify mysql result 2022-04-29 16:59:49 +08:00
as2252258 cf22354c4e modify mysql result 2022-04-29 16:58:55 +08:00
as2252258 2415e24045 modify mysql result 2022-04-29 16:57:39 +08:00
as2252258 7e2d920f6c modify mysql result 2022-04-29 16:54:33 +08:00
as2252258 c7760fad12 modify mysql result 2022-04-29 16:53:11 +08:00
as2252258 ef3ea8c7a4 modify mysql result 2022-04-29 16:51:27 +08:00
as2252258 91aae64512 modify mysql result 2022-04-29 14:49:29 +08:00
as2252258 af768725d8 modify mysql result 2022-04-29 14:46:45 +08:00
as2252258 021076cf51 modify mysql result 2022-04-28 11:56:19 +08:00
as2252258 4357ac890a modify mysql result 2022-04-27 13:48:11 +08:00
as2252258 d8005f21f9 modify mysql result 2022-04-10 15:34:12 +08:00
as2252258 ac78165c25 modify mysql result 2022-04-10 15:13:03 +08:00
as2252258 96f8b6e9c9 modify mysql result 2022-04-10 14:58:35 +08:00
as2252258 4887a4c023 modify mysql result 2022-04-10 14:56:32 +08:00
as2252258 8a4be3ac97 modify mysql result 2022-04-10 14:54:46 +08:00
as2252258 9985f51e52 modify mysql result 2022-04-10 14:31:40 +08:00
as2252258 01458e80f0 modify mysql result 2022-04-10 14:28:35 +08:00
as2252258 93e1b49a8c modify mysql result 2022-04-10 14:27:39 +08:00
as2252258 e56139b39e modify mysql result 2022-04-10 03:55:49 +08:00
as2252258 536821d95d modify mysql result 2022-04-10 03:53:21 +08:00
as2252258 b56978e65c modify mysql result 2022-04-10 03:41:10 +08:00
as2252258 f4a0056e76 modify mysql result 2022-04-10 03:40:02 +08:00
as2252258 f091fd5ba0 modify mysql result 2022-04-10 03:37:58 +08:00
as2252258 8037bfa1b9 modify plugin name 2022-03-03 18:11:24 +08:00
as2252258 84eac26247 modify plugin name 2022-03-03 18:06:18 +08:00
as2252258 e2a6ce6981 modify plugin name 2022-02-28 17:42:28 +08:00
as2252258 d9742f5579 modify plugin name 2022-02-27 16:06:14 +08:00
as2252258 046ca468d4 modify plugin name 2022-02-23 18:12:56 +08:00
as2252258 cf50859f30 modify plugin name 2022-02-23 18:02:28 +08:00
as2252258 c4c68d8c3c modify plugin name 2022-02-23 16:42:59 +08:00
as2252258 2157ac8174 modify plugin name 2022-02-23 16:32:07 +08:00
as2252258 0f693e3b37 modify plugin name 2022-02-22 11:47:16 +08:00
as2252258 d44923d477 modify plugin name 2022-02-21 11:10:30 +08:00
as2252258 8497df9559 modify plugin name 2022-02-21 11:02:10 +08:00
as2252258 1059537fa7 modify plugin name 2022-02-21 10:51:38 +08:00
as2252258 541df13de2 modify plugin name 2022-02-18 15:28:28 +08:00
as2252258 3924139407 modify plugin name 2022-02-18 15:16:40 +08:00
as2252258 297990dd34 modify plugin name 2022-02-18 14:22:26 +08:00
as2252258 7e87d86546 modify plugin name 2022-02-18 14:01:24 +08:00
as2252258 28d3d48ab0 modify plugin name 2022-02-17 17:44:27 +08:00
as2252258 e26e16ba26 modify plugin name 2022-02-15 17:25:14 +08:00
as2252258 81f3278dd7 modify plugin name 2022-02-15 11:16:53 +08:00
as2252258 7b50598783 modify plugin name 2022-02-14 18:51:07 +08:00
as2252258 4f366fe0e6 modify plugin name 2022-02-14 18:40:22 +08:00
as2252258 209cc8b507 modify plugin name 2022-02-14 18:39:54 +08:00
as2252258 4cd5d0fbf1 modify plugin name 2022-02-14 18:05:50 +08:00
as2252258 07a908840a modify plugin name 2022-02-14 17:44:34 +08:00
as2252258 f4cc6d6a23 modify plugin name 2022-02-14 17:21:12 +08:00
as2252258 97519bda36 modify plugin name 2022-02-14 16:59:39 +08:00
as2252258 8f482ea3b6 modify plugin name 2022-02-14 16:04:09 +08:00
as2252258 e094733d7b modify plugin name 2022-02-14 14:47:53 +08:00
as2252258 7a0322d8b3 modify plugin name 2022-02-14 14:46:41 +08:00
as2252258 9d717b0f7a modify plugin name 2022-02-14 14:42:56 +08:00
as2252258 a4bd152b59 modify plugin name 2022-02-14 14:42:07 +08:00
as2252258 12a401be8e modify plugin name 2022-02-14 14:41:41 +08:00
as2252258 485b951e52 modify plugin name 2022-02-14 14:40:38 +08:00
as2252258 2e3246dc23 modify plugin name 2022-02-14 14:34:56 +08:00
as2252258 7999847ffa modify plugin name 2022-02-14 14:28:46 +08:00
as2252258 f272d8e451 modify plugin name 2022-02-14 11:31:25 +08:00
as2252258 fed145a43f modify plugin name 2022-02-14 11:30:06 +08:00
as2252258 b6f7b2622d modify plugin name 2022-02-14 11:28:45 +08:00
as2252258 1bdbca2568 modify plugin name 2022-02-14 10:55:56 +08:00
as2252258 2507c30b27 modify plugin name 2022-02-14 10:45:39 +08:00
as2252258 567ceb69c0 modify plugin name 2022-02-14 10:42:27 +08:00
as2252258 f1a9b74122 modify plugin name 2022-02-13 03:15:05 +08:00
as2252258 dc47bd5106 modify plugin name 2022-02-11 19:00:55 +08:00
as2252258 0bf995f3e4 modify plugin name 2022-02-11 16:44:10 +08:00
as2252258 c2fde2e6d7 Revert "改名"
This reverts commit fdf58326
2022-01-24 11:46:11 +08:00
as2252258 5ddd0bc6d7 Revert "改名"
This reverts commit fdf58326
2022-01-21 15:22:32 +08:00
as2252258 d22db391f3 Revert "改名"
This reverts commit fdf58326
2022-01-20 19:04:15 +08:00
as2252258 bcd78aba56 Revert "改名"
This reverts commit fdf58326
2022-01-18 17:50:32 +08:00
as2252258 c899f51f35 Revert "改名"
This reverts commit fdf58326
2022-01-18 16:07:05 +08:00
as2252258 1419e96c89 Revert "改名"
This reverts commit fdf58326
2022-01-17 19:04:26 +08:00
as2252258 224b52db49 Revert "改名"
This reverts commit fdf58326
2022-01-17 18:45:00 +08:00
as2252258 248f4d7100 Revert "改名"
This reverts commit fdf58326
2022-01-14 16:11:23 +08:00
as2252258 122f37d6ce Revert "改名"
This reverts commit fdf58326
2022-01-14 15:23:29 +08:00
as2252258 9fe40ff78a Revert "改名"
This reverts commit fdf58326
2022-01-14 15:20:57 +08:00
as2252258 ae4d75f310 Revert "改名"
This reverts commit fdf58326
2022-01-14 14:53:38 +08:00
as2252258 707379c0fd Revert "改名"
This reverts commit fdf58326
2022-01-14 14:52:56 +08:00
as2252258 b861e0b17b Revert "改名"
This reverts commit fdf58326
2022-01-14 14:51:35 +08:00
as2252258 7b576476a2 Revert "改名"
This reverts commit fdf58326
2022-01-14 14:41:57 +08:00
as2252258 711a819ebb Revert "改名"
This reverts commit fdf58326
2022-01-14 14:40:54 +08:00
as2252258 971b1f1fb0 Revert "改名"
This reverts commit fdf58326
2022-01-14 14:18:28 +08:00
as2252258 4831bc67f5 Revert "改名"
This reverts commit fdf58326
2022-01-14 11:29:15 +08:00
as2252258 1b8c6ecde0 Revert "改名"
This reverts commit fdf58326
2022-01-13 18:55:02 +08:00
as2252258 a4c78874e4 Revert "改名"
This reverts commit fdf58326
2022-01-13 18:41:15 +08:00
as2252258 b6c4693ef6 Revert "改名"
This reverts commit fdf58326
2022-01-13 18:36:25 +08:00
as2252258 adfa1cf3f2 Revert "改名"
This reverts commit fdf58326
2022-01-12 16:00:11 +08:00
as2252258 6362a5ce91 Revert "改名"
This reverts commit fdf58326
2022-01-12 14:54:51 +08:00
as2252258 edeedd2258 Revert "改名"
This reverts commit fdf58326
2022-01-12 14:46:10 +08:00
as2252258 02d0c9a8fd Revert "改名"
This reverts commit fdf58326
2022-01-12 14:43:28 +08:00
as2252258 d707b7f384 Revert "改名"
This reverts commit fdf58326
2022-01-12 14:10:32 +08:00
as2252258 b6f1d8eaf7 Revert "改名"
This reverts commit fdf58326
2022-01-12 11:52:58 +08:00
as2252258 c67fc70550 Revert "改名"
This reverts commit fdf58326
2022-01-12 11:42:30 +08:00
as2252258 e55be953bf Revert "改名"
This reverts commit fdf58326
2022-01-12 11:41:26 +08:00
as2252258 aca8cac1a4 Revert "改名"
This reverts commit fdf58326
2022-01-12 11:39:16 +08:00
as2252258 e699907970 Revert "改名"
This reverts commit fdf58326
2022-01-12 11:32:36 +08:00
as2252258 97dc2f559b Revert "改名"
This reverts commit fdf58326
2022-01-12 11:29:50 +08:00
as2252258 33a01eda28 Revert "改名"
This reverts commit fdf58326
2022-01-12 11:20:33 +08:00
as2252258 b38704374a Revert "改名"
This reverts commit fdf58326
2022-01-11 17:58:04 +08:00
as2252258 318377212c Revert "改名"
This reverts commit fdf58326
2022-01-11 16:16:04 +08:00
as2252258 ddbdae5013 Revert "改名"
This reverts commit fdf58326
2022-01-11 10:26:34 +08:00
as2252258 6eb7aa1fb0 Revert "改名"
This reverts commit fdf58326
2022-01-10 19:03:55 +08:00
as2252258 078e8338fe Revert "改名"
This reverts commit fdf58326
2022-01-10 18:57:54 +08:00
as2252258 6bcd76333a Revert "改名"
This reverts commit fdf58326
2022-01-10 13:43:01 +08:00
as2252258 b2728947a3 Revert "改名"
This reverts commit fdf58326
2022-01-10 11:39:55 +08:00
as2252258 8ec5b0e8d8 1 2022-01-09 17:56:47 +08:00
as2252258 2939d58585 1 2022-01-09 15:44:54 +08:00
as2252258 146da1d00c 1 2022-01-09 15:41:37 +08:00
as2252258 51babeddc1 1 2022-01-09 03:55:58 +08:00
as2252258 550197de96 e 2022-01-09 03:49:02 +08:00
as2252258 b5ad99332d 1 2022-01-09 03:46:41 +08:00
as2252258 e9b51ffb1e 1 2022-01-09 02:44:05 +08:00
as2252258 8c9de79027 Revert "改名"
This reverts commit fdf58326
2022-01-08 18:49:06 +08:00
as2252258 85c973469f Revert "改名"
This reverts commit fdf58326
2022-01-08 10:07:19 +08:00
as2252258 db8d9a8376 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:58:56 +08:00
as2252258 18b4e79f5f Revert "改名"
This reverts commit fdf58326
2022-01-07 18:58:16 +08:00
as2252258 0b7f380298 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:56:58 +08:00
as2252258 ea400c4ea5 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:54:46 +08:00
as2252258 07cfb62bb8 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:54:18 +08:00
as2252258 a5efa0813f Revert "改名"
This reverts commit fdf58326
2022-01-07 18:51:46 +08:00
as2252258 d1f2912324 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:51:12 +08:00
as2252258 ba0acf8f00 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:50:24 +08:00
as2252258 2788008fa1 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:45:44 +08:00
as2252258 5beee8352d Revert "改名"
This reverts commit fdf58326
2022-01-07 18:41:28 +08:00
as2252258 35859fead1 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:40:45 +08:00
as2252258 0031d73eff Revert "改名"
This reverts commit fdf58326
2022-01-07 18:40:05 +08:00
as2252258 89807fe548 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:38:45 +08:00
as2252258 9c28ce91ae Revert "改名"
This reverts commit fdf58326
2022-01-07 18:35:35 +08:00
as2252258 7ed01ba893 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:33:31 +08:00
as2252258 926a8c0719 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:29:08 +08:00
as2252258 688a30531a Revert "改名"
This reverts commit fdf58326
2022-01-07 18:27:01 +08:00
as2252258 1ff9d7f210 Revert "改名"
This reverts commit fdf58326
2022-01-07 18:16:29 +08:00
as2252258 da038f4fd8 Revert "改名"
This reverts commit fdf58326
2022-01-07 14:38:36 +08:00
as2252258 de20fa9582 Revert "改名"
This reverts commit fdf58326
2022-01-06 19:05:15 +08:00
as2252258 80cf72adc9 Revert "改名"
This reverts commit fdf58326
2022-01-04 16:06:29 +08:00
as2252258 dabcea452a 改名 2021-12-23 18:25:31 +08:00
as2252258 2feda943af 改名 2021-12-23 18:22:26 +08:00
as2252258 bffe375230 1 2021-12-17 04:20:20 +08:00
as2252258 efb69d02f3 改名 2021-12-08 11:46:08 +08:00
as2252258 79e5fa0b08 改名 2021-12-08 11:45:19 +08:00
as2252258 de7e005b46 改名 2021-12-08 11:32:31 +08:00
as2252258 a64fa8f493 改名 2021-12-08 11:16:35 +08:00
as2252258 c19e141f68 改名 2021-12-06 17:58:11 +08:00
as2252258 94a522588d 改名 2021-12-06 13:51:54 +08:00
as2252258 308c992fd3 改名 2021-12-06 11:47:12 +08:00
as2252258 458c6cbd22 改名 2021-12-03 16:05:49 +08:00
as2252258 b717a7e9f7 改名 2021-12-03 15:42:04 +08:00
as2252258 6786b313b5 改名 2021-12-03 14:46:38 +08:00
as2252258 14db451313 改名 2021-12-02 18:24:23 +08:00
as2252258 ced92a10e1 改名 2021-12-02 14:16:57 +08:00
as2252258 0d12baf49f 改名 2021-12-01 19:04:59 +08:00
as2252258 b798270bb7 改名 2021-12-01 17:21:15 +08:00
as2252258 955f358577 改名 2021-12-01 16:09:49 +08:00
as2252258 97a5deda23 改名 2021-12-01 15:28:59 +08:00
as2252258 8ae395e9cd 改名 2021-12-01 15:18:34 +08:00
as2252258 44a9a507f6 改名 2021-12-01 15:16:08 +08:00
as2252258 903ee0d6ce 改名 2021-11-30 16:00:39 +08:00
as2252258 a36943cd2d 改名 2021-11-30 15:10:00 +08:00
as2252258 26e0ce7778 改名 2021-11-30 14:59:51 +08:00
as2252258 5a9f9da347 改名 2021-11-30 14:43:55 +08:00
as2252258 a1bf157408 改名 2021-11-30 14:34:15 +08:00
as2252258 27834d37ed 改名 2021-11-30 14:32:56 +08:00
as2252258 c260ac6df4 1 2021-11-27 17:43:28 +08:00
57 changed files with 2655 additions and 1922 deletions
+235
View File
@@ -0,0 +1,235 @@
<?php
namespace Kiri\Server\Abstracts;
use Exception;
use Kiri\Exception\ConfigException;
use Kiri\Exception\NotFindClassException;
use Kiri\Server\Config as SConfig;
use Kiri\Server\Constant;
use Kiri\Server\Events\OnServerBeforeStart;
use Kiri\Server\Events\OnShutdown;
use Kiri\Server\Handler\OnServer;
use Kiri\Server\ServerInterface;
use Kiri\Server\Task\Task;
use Kiri\Di\Inject\Container;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\ContainerInterface;
use Psr\Container\NotFoundExceptionInterface;
use ReflectionException;
use Swoole\Server;
use Symfony\Component\Console\Output\OutputInterface;
/**
*
*/
class AsyncServer implements ServerInterface
{
use TraitServer;
/**
* @var Server|null
*/
private ?Server $server = null;
/**
* @var ContainerInterface
*/
#[Container(ContainerInterface::class)]
public ContainerInterface $container;
/**
* @param array $service
* @param int $daemon
* @return void
* @throws
*/
public function initCoreServers(array $service, int $daemon = 0): void
{
$service = $this->genConfigService($service);
$this->createBaseServer(array_pop($service), $daemon);
foreach ($service as $value) {
$this->addListener($value);
}
foreach ($this->_process as $process) {
$this->server->addProcess($process);
}
on(OnServerBeforeStart::class, [$this, 'onSignal']);
}
/**
* @return bool
* @throws
*/
public function shutdown(): bool
{
$this->server->shutdown();
event(new OnShutdown());
return true;
}
/**
* @param SConfig $config
* @param int $daemon
* @return void
* @throws
*/
private function createBaseServer(SConfig $config, int $daemon = 0): void
{
$match = $this->getServerClass($config->type);
if (is_null($match)) {
throw new NotFindClassException('Unknown server type ' . $config->type);
}
$this->initServer($match, $config, $daemon);
$this->onEventListen($this->server, \config('server.events', []));
$this->onEventListen($this->server, $config->events);
$this->onTaskListen();
}
/**
* @param $match
* @param $config
* @param $daemon
* @return void
* @throws
*/
private function initServer($match, $config, $daemon): void
{
$this->server = new $match($config->host, $config->port, $config->mode, $config->socket);
$this->server->set($this->systemConfig($config, $daemon));
if (!isset($config->events[Constant::SHUTDOWN])) {
$config->events[Constant::SHUTDOWN] = [OnServer::class, 'onShutdown'];
}
$this->_listenDump($config);
$this->container->bind(ServerInterface::class, $this->server);
}
/**
* @return void
* @throws
*/
private function onTaskListen(): void
{
if (!isset($this->server->setting[Constant::OPTION_TASK_WORKER_NUM])) {
return;
}
$this->container->get(Task::class)->initTaskWorker($this->server);
}
/**
* @param SConfig $config
* @param int $daemon
* @return array
* @throws
*/
protected function systemConfig(SConfig $config, int $daemon): array
{
$settings = array_merge(\config('server.settings', []), $config->settings);
$settings[Constant::OPTION_DAEMONIZE] = (bool)$daemon;
$settings[Constant::OPTION_ENABLE_REUSE_PORT] = true;
$settings[Constant::OPTION_PID_FILE] = storage('.swoole.pid');
if (!isset($settings[Constant::OPTION_PID_FILE])) {
$settings[Constant::OPTION_LOG_FILE] = storage('system.log');
}
return $settings;
}
/**
* @param SConfig $config
* @return void
* @throws
*/
public function addListener(SConfig $config): void
{
$port = $this->server->addlistener($config->host, $config->port, $config->socket);
if ($port === false) {
throw new Exception('Listen port fail.' . swoole_last_error());
}
$this->_listenDump($config);
$port->set($this->resetSettings($config->type, $config->settings));
$this->onEventListen($port, $config->getEvents());
}
/**
* @param SConfig $config
* @return void
* @throws
*/
protected function _listenDump(SConfig $config): void
{
$writeln = $this->container->get(OutputInterface::class);
if ($config->type == Constant::SERVER_TYPE_HTTP) {
$writeln->writeln('Add http port listen ' . $config->host . '::' . $config->port);
} else if ($config->type == Constant::SERVER_TYPE_WEBSOCKET) {
$writeln->writeln('Add wss port listen ' . $config->host . '::' . $config->port);
} else if ($config->type == Constant::SERVER_TYPE_UDP) {
$writeln->writeln('Add udp port listen ' . $config->host . '::' . $config->port);
} else {
$writeln->writeln('Add tcp port listen ' . $config->host . '::' . $config->port);
}
}
/**
* @param string $type
* @param array $settings
* @return array
* @throws
*/
private function resetSettings(string $type, array $settings): array
{
if ($type == Constant::SERVER_TYPE_HTTP && !isset($settings['open_http_protocol'])) {
$settings['open_http_protocol'] = true;
if (in_array($this->server->setting['dispatch_mode'], [2, 4])) {
$settings['open_http2_protocol'] = true;
}
}
if ($type == Constant::SERVER_TYPE_WEBSOCKET && !isset($settings['open_websocket_protocol'])) {
$settings['open_websocket_protocol'] = true;
}
return $settings;
}
/**
* @param Server\Port|Server $base
* @param array $events
* @return void
* @throws
*/
private function onEventListen(Server\Port|Server $base, array $events): void
{
foreach ($events as $name => $event) {
if (is_array($event) && is_string($event[0])) {
$event[0] = $this->container->get($event[0]);
}
$base->on($name, $event);
}
}
/**
* @return void
* @throws
*/
public function start(): void
{
event(new OnServerBeforeStart());
$this->server->start();
}
}
+151 -83
View File
@@ -1,83 +1,151 @@
<?php
namespace Server\Abstracts;
use JetBrains\PhpStorm\Pure;
use Server\Contract\OnProcessInterface;
use Swoole\Coroutine;
use Swoole\Process;
/**
*
*/
abstract class BaseProcess implements OnProcessInterface
{
protected bool $isStop = false;
protected mixed $redirect_stdin_and_stdout = null;
protected int $pipe_type = SOCK_DGRAM;
protected bool $enable_coroutine = true;
public string $name = 'swoole process.';
/**
* @return string
*/
public function getName(): string
{
return $this->name;
}
/**
* @return bool
*/
public function isStop(): bool
{
return $this->isStop;
}
/**
* @return mixed
*/
public function getRedirectStdinAndStdout(): mixed
{
return $this->redirect_stdin_and_stdout;
}
/**
* @return int
*/
public function getPipeType(): int
{
return $this->pipe_type;
}
/**
* @return bool
*/
public function isEnableCoroutine(): bool
{
return $this->enable_coroutine;
}
/**
*
*/
public function onProcessStop(): void
{
$this->isStop = true;
}
}
<?php
namespace Kiri\Server\Abstracts;
use Kiri\Di\Context;
use Kiri\Di\Inject\Container;
use Kiri\Error\StdoutLogger;
use Kiri\Server\Contract\OnProcessInterface;
use Psr\Container\ContainerInterface;
use Psr\Log\LoggerInterface;
use Swoole\Coroutine;
/**
*
*/
abstract class BaseProcess implements OnProcessInterface
{
private bool $stop = false;
/**
* @var bool
*/
protected bool $redirect_stdin_and_stdout = FALSE;
/**
* @var int
*/
protected int $pipe_type = SOCK_DGRAM;
/**
* @var bool
*/
protected bool $enable_coroutine = false;
/**
* @var bool
*/
protected bool $enable_queue = false;
/**
* @var StdoutLogger
*/
#[Container(LoggerInterface::class)]
public StdoutLogger $logger;
/**
* @var \Kiri\Di\Container
*/
#[Container(ContainerInterface::class)]
public \Kiri\Di\Container $container;
/**
* @var string
*/
public string $name = '';
/**
* @return bool
*/
public function isEnableQueue(): bool
{
return $this->enable_queue;
}
/**
* @return string
*/
public function getName(): string
{
if (empty($this->name)) {
$this->name = uniqid('p.');
}
return $this->name;
}
/**
* @return bool
*/
public function isStop(): bool
{
return $this->stop;
}
/**
* @return bool
*/
public function getRedirectStdinAndStdout(): bool
{
return $this->redirect_stdin_and_stdout;
}
/**
* @return int
*/
public function getPipeType(): int
{
return $this->pipe_type;
}
/**
* @return bool
*/
public function isEnableCoroutine(): bool
{
return $this->enable_coroutine;
}
/**
*
*/
public function stop(): void
{
$this->stop = true;
}
/**
* @return $this
*/
abstract public function onSigterm(): static;
/**
* @param $data
* @return void
*/
protected function onShutdown($data): void
{
$this->stop = true;
$value = Context::get('waite:process:message');
$this->logger->alert('Process ' . $this->getName() . ' stop');
if (!is_null($value) && Coroutine::exists((int)$value)) {
Coroutine::cancel((int)$value);
}
}
}
+22 -55
View File
@@ -1,55 +1,22 @@
<?php
namespace Server\Abstracts;
use Annotation\Inject;
use Exception;
use Kiri\Abstracts\Config;
use Kiri\Exception\ConfigException;
use Kiri\Kiri;
use Psr\Log\LoggerInterface;
/**
* Class Server
* @package Server\Abstracts
*/
abstract class Server
{
/**
* @var LoggerInterface
*/
#[Inject(LoggerInterface::class)]
public LoggerInterface $logger;
/**
* @param $prefix
* @throws ConfigException
*/
protected function setProcessName($prefix)
{
if (Kiri::getPlatform()->isMac()) {
return;
}
$name = '[' . Config::get('id', 'system-service') . ']';
if (!empty($prefix)) {
$name .= '.' . $prefix;
}
swoole_set_process_name($name);
}
/**
* Server constructor.
* @throws Exception
*/
public function __construct()
{
}
}
<?php
namespace Kiri\Server\Abstracts;
/**
* Class Server
* @package Server\Abstracts
*/
abstract class Server
{
/**
* Server constructor.
* @throws
*/
public function __construct()
{
}
}
+209
View File
@@ -0,0 +1,209 @@
<?php
namespace Kiri\Server\Abstracts;
use Exception;
use Kiri;
use Kiri\Di\Inject\Container;
use Psr\Log\LoggerInterface;
use Swoole\Http\Server as HServer;
use Swoole\Process;
use Swoole\Server;
use Kiri\Server\Constant;
use Kiri\Server\Config;
use Kiri\Error\StdoutLogger;
use Swoole\WebSocket\Server as WServer;
trait TraitServer
{
/**
* @var array
*/
private array $_process = [];
/**
* @var StdoutLogger
*/
#[Container(LoggerInterface::class)]
public StdoutLogger $logger;
/**
* @param string|array|BaseProcess $class
* @return void
* @throws
*/
public function addProcess(string|array|BaseProcess $class): void
{
if (!is_array($class)) {
$class = [$class];
}
foreach ($class as $name) {
if (is_string($name)) {
$name = Kiri::getDi()->get($name);
}
if (isset($this->_process[$name->getName()])) {
throw new Exception('Process(' . $name->getName() . ') is exists.');
}
$process = $this->genProcess($name);
if ($name->isEnableQueue()) {
$process->useQueue();
}
$this->_process[$name->getName()] = $process;
}
}
/**
* @param BaseProcess $name
* @return Process
*/
private function genProcess(BaseProcess $name): Process
{
return new Process(function (Process $process) use ($name) {
$process->name('[' . \config('id','system-service') . ']' . $name->getName());
$name->onSigterm()->process($process);
},
$name->getRedirectStdinAndStdout(),
$name->getPipeType(),
$name->isEnableCoroutine());
}
/**
* @param string $name
* @return Process|null
*/
public function getProcess(string $name): ?Process
{
return $this->_process[$name] ?? null;
}
/**
* @return void
* @throws
*/
public function onSignal(): void
{
$signal = \config('signal', []);
$this->onPcntlSignal(SIGINT, [$this, 'onSigint']);
foreach ($signal as $sig => $value) {
if (is_array($value) && is_string($value[0])) {
$value[0] = \Kiri::getDi()->get($value[0]);
}
if (!is_callable($value, true)) {
throw new Exception('Register signal callback must can exec.');
}
$this->onPcntlSignal($sig, $value);
}
}
/**
* @param $no
* @param array $signInfo
* @return void
*/
public function onSigint($no, array $signInfo): void
{
try {
$this->logger->alert('Pid ' . getmypid() . ' get signo ' . $no);
$this->shutdown();
} catch (\Throwable $exception) {
error($exception);
}
}
/**
* @param $signal
* @param $callback
* @return void
*/
private function onPcntlSignal($signal, $callback): void
{
\pcntl_signal($signal, $callback);
}
/**
* @return array
*/
public function getProcesses(): array
{
return $this->_process;
}
/**
* @param array $ports
* @return array
*/
public function sortService(array $ports): array
{
$array = [];
foreach ($ports as $port) {
$array = $this->sort($array, $port);
}
return $array;
}
/**
* @param array $ports
* @return array<Config>
*/
public function genConfigService(array $ports): array
{
$array = [];
$ports = $ports['ports'] ?? [];
foreach ($ports as $port) {
$array = $this->sort($array, $port);
}
return $array;
}
/**
* @param array $array
* @param $port
* @return array
*/
private function sort(array $array, $port): array
{
$config = instance(Config::class, [], $port);
if ($port['type'] == Constant::SERVER_TYPE_WEBSOCKET) {
array_unshift($array, $config);
} else if ($port['type'] == Constant::SERVER_TYPE_HTTP) {
if (!empty($array) && $array[0]['type'] == Constant::SERVER_TYPE_WEBSOCKET) {
$array[] = $config;
} else {
array_unshift($array, $config);
}
} else {
$array[] = $config;
}
return $array;
}
/**
* @param $type
* @return string|null
*/
public function getServerClass($type): ?string
{
return match ($type) {
Constant::SERVER_TYPE_BASE, Constant::SERVER_TYPE_TCP,
Constant::SERVER_TYPE_UDP => Server::class,
Constant::SERVER_TYPE_HTTP => HServer::class,
Constant::SERVER_TYPE_WEBSOCKET => WServer::class,
default => null
};
}
}
+163
View File
@@ -0,0 +1,163 @@
<?php
namespace Kiri\Server;
/**
*
*/
class Config
{
public string $type;
public string $host = '';
public int $port = 0;
public string $name = '';
public int $mode = SWOOLE_PROCESS;
public int $socket = SWOOLE_SOCK_TCP;
public array $settings = [];
public array $events = [];
/**
* @return string
*/
public function getType(): string
{
return $this->type;
}
/**
* @param string $type
*/
public function setType(string $type): void
{
$this->type = $type;
}
/**
* @return string
*/
public function getHost(): string
{
return $this->host;
}
/**
* @param string $host
*/
public function setHost(string $host): void
{
$this->host = $host;
}
/**
* @return int
*/
public function getPort(): int
{
return $this->port;
}
/**
* @param int $port
*/
public function setPort(int $port): void
{
$this->port = $port;
}
/**
* @return string
*/
public function getName(): string
{
return $this->name;
}
/**
* @param string $name
*/
public function setName(string $name): void
{
$this->name = $name;
}
/**
* @return int
*/
public function getMode(): int
{
return $this->mode;
}
/**
* @param int $mode
*/
public function setMode(int $mode): void
{
$this->mode = $mode;
}
/**
* @return array
*/
public function getSettings(): array
{
return $this->settings;
}
/**
* @param array $settings
*/
public function setSettings(array $settings): void
{
$this->settings = $settings;
}
/**
* @return array
*/
public function getEvents(): array
{
return $this->events;
}
/**
* @param array $events
*/
public function setEvents(array $events): void
{
$this->events = $events;
}
/**
* @return int
*/
public function getSocket(): int
{
return $this->socket;
}
/**
* @param int $socket
*/
public function setSocket(int $socket): void
{
$this->socket = $socket;
}
}
+133 -133
View File
@@ -1,133 +1,133 @@
<?php
namespace Server;
/**
* Class Constant
* @package Server
*/
class Constant
{
const START = 'Start';
const SHUTDOWN = 'Shutdown';
const BEFORE_SHUTDOWN = 'beforeShutdown';
const WORKER_START = 'WorkerStart';
const WORKER_STOP = 'WorkerStop';
const WORKER_EXIT = 'WorkerExit';
const CONNECT = 'Connect';
const HANDSHAKE = 'handshake';
const OPEN = 'open';
const DISCONNECT = 'disconnect';
const MESSAGE = 'message';
const RECEIVE = 'Receive';
const PACKET = 'Packet';
const REQUEST = 'request';
const CLOSE = 'Close';
const TASK = 'Task';
const FINISH = 'Finish';
const PIPE_MESSAGE = 'PipeMessage';
const WORKER_ERROR = 'WorkerError';
const MANAGER_START = 'ManagerStart';
const MANAGER_STOP = 'ManagerStop';
const BEFORE_RELOAD = 'BeforeReload';
const AFTER_RELOAD = 'AfterReload';
const SERVER_TYPE_HTTP = 'http';
const SERVER_TYPE_WEBSOCKET = 'ws';
const SERVER_TYPE_TCP = 'tcp';
const SERVER_TYPE_UDP = 'udp';
const SERVER_TYPE_BASE = 'base';
const STATUS_404_MESSAGE = '<h2>HTTP 404 Not Found</h2><hr><i>Powered by Swoole</i>';
const STATUS_405_MESSAGE = '<h2>HTTP 405 Method allow</h2><hr><i>Powered by Swoole</i>';
const OPTION_REACTOR_NUM = 'reactor_num';
const OPTION_WORKER_NUM = 'worker_num';
const OPTION_MAX_REQUEST = 'max_request';
const OPTION_MAX_CONN = 'max_connection';
const OPTION_TASK_WORKER_NUM = 'task_worker_num';
const OPTION_TASK_IPC_MODE = 'task_ipc_mode';
const OPTION_TASK_MAX_REQUEST = 'task_max_request';
const OPTION_TASK_TMPDIR = 'task_tmpdir';
const OPTION_TASK_ENABLE_COROUTINE = 'task_enable_coroutine';
const OPTION_TASK_USE_OBJECT = 'task_use_object';
const OPTION_DISPATCH_MODE = 'dispatch_mode';
const OPTION_DISPATCH_FUNC = 'dispatch_func';
const OPTION_MESSAGE_QUEUE_KEY = 'message_queue_key';
const OPTION_DAEMONIZE = 'daemonize';
const OPTION_BACKLOG = 'backlog';
const OPTION_LOG_FILE = 'log_file';
const OPTION_LOG_LEVEL = 'log_level';
const OPTION_LOG_DATE_WITH_MICROSECONDS = 'log_date_with_microseconds';
const OPTION_LOG_ROTATION = 'log_rotation';
const OPTION_LOG_DATE_FORMAT = 'log_date_format';
const OPTION_OPEN_TCP_KEEPALIVE = 'open_tcp_keepalive';
const OPTION_HEARTBEAT_CHECK_INTERVAL = 'heartbeat_check_interval';
const OPTION_HEARTBEAT_IDLE_TIME = 'heartbeat_idle_time';
const OPTION_OPEN_EOF_CHECK = 'open_eof_check';
const OPTION_OPEN_EOF_SPLIT = 'open_eof_split';
const OPTION_PACKAGE_EOF = 'package_eof';
const OPTION_OPEN_LENGTH_CHECK = 'open_length_check';
const OPTION_PACKAGE_LENGTH_TYPE = 'package_length_type';
const OPTION_PACKAGE_LENGTH_FUNC = 'package_length_func';
const OPTION_PACKAGE_MAX_LENGTH = 'package_max_length';
const OPTION_OPEN_HTTP_PROTOCOL = 'open_http_protocol';
const OPTION_OPEN_MQTT_PROTOCOL = 'open_mqtt_protocol';
const OPTION_OPEN_REDIS_PROTOCOL = 'open_redis_protocol';
const OPTION_OPEN_WEBSOCKET_PROTOCOL = 'open_websocket_protocol';
const OPTION_OPEN_WEBSOCKET_CLOSE_FRAME = 'open_websocket_close_frame';
const OPTION_OPEN_TCP_NODELAY = 'open_tcp_nodelay';
const OPTION_OPEN_CPU_AFFINITY = 'open_cpu_affinity';
const OPTION_CPU_AFFINITY_IGNORE = 'cpu_affinity_ignore';
const OPTION_TCP_DEFER_ACCEPT = 'tcp_defer_accept';
const OPTION_SSL_CERT_FILE = 'ssl_cert_file';
const OPTION_SSL_KEY_FILE = 'ssl_key_file';
const OPTION_SSL_METHOD = 'ssl_method';
const OPTION_SSL_PROTOCOLS = 'ssl_protocols';
const OPTION_SSL_SNI_CERTS = 'ssl_sni_certs';
const OPTION_SSL_CIPHERS = 'ssl_ciphers';
const OPTION_SSL_VERIFY_PEER = 'ssl_verify_peer';
const OPTION_SSL_ALLOW_SELF_SIGNED = 'ssl_allow_self_signed';
const OPTION_SSL_CLIENT_CERT_FILE = 'ssl_client_cert_file';
const OPTION_SSL_COMPRESS = 'ssl_compress';
const OPTION_SSL_VERIFY_DEPTH = 'ssl_verify_depth';
const OPTION_SSL_PREFER_SERVER_CIPHERS = 'ssl_prefer_server_ciphers';
const OPTION_SSL_DHPARAM = 'ssl_dhparam';
const OPTION_SSL_ECDH_CURVE = 'ssl_ecdh_curve';
const OPTION_USER = 'user';
const OPTION_GROUP = 'group';
const OPTION_CHROOT = 'chroot';
const OPTION_PID_FILE = 'pid_file';
const OPTION_BUFFER_INPUT_SIZE = 'buffer_input_size';
const OPTION_BUFFER_OUTPUT_SIZE = 'buffer_output_size';
const OPTION_SOCKET_BUFFER_SIZE = 'socket_buffer_size';
const OPTION_ENABLE_UNSAFE_EVENT = 'enable_unsafe_event';
const OPTION_DISCARD_TIMEOUT_REQUEST = 'discard_timeout_request';
const OPTION_ENABLE_REUSE_PORT = 'enable_reuse_port';
const OPTION_ENABLE_DELAY_RECEIVE = 'enable_delay_receive';
const OPTION_RELOAD_ASYNC = 'reload_async';
const OPTION_MAX_WAIT_TIME = 'max_wait_time';
const OPTION_TCP_FASTOPEN = 'tcp_fastopen';
const OPTION_REQUEST_SLOWLOG_FILE = 'request_slowlog_file';
const OPTION_ENABLE_COROUTINE = 'enable_coroutine';
const OPTION_MAX_COROUTINE = 'max_coroutine';
const OPTION_SEND_YIELD = 'send_yield';
const OPTION_SEND_TIMEOUT = 'send_timeout';
const OPTION_HOOK_FLAGS = 'hook_flags';
const OPTION_BUFFER_HIGH_WATERMARK = 'buffer_high_watermark';
const OPTION_BUFFER_LOW_WATERMARK = 'buffer_low_watermark';
const OPTION_TCP_USER_TIMEOUT = 'tcp_user_timeout';
const OPTION_STATS_FILE = 'stats_file';
const OPTION_EVENT_OBJECT = 'event_object';
const OPTION_START_SESSION_ID = 'start_session_id';
const OPTION_SINGLE_THREAD = 'single_thread';
const OPTION_MAX_QUEUED_BYTES = 'max_queued_bytes';
}
<?php
namespace Kiri\Server;
/**
* Class Constant
* @package Server
*/
class Constant
{
const string START = 'Start';
const string SHUTDOWN = 'Shutdown';
const string BEFORE_SHUTDOWN = 'beforeShutdown';
const string WORKER_START = 'WorkerStart';
const string WORKER_STOP = 'WorkerStop';
const string WORKER_EXIT = 'WorkerExit';
const string CONNECT = 'Connect';
const string HANDSHAKE = 'handshake';
const string OPEN = 'open';
const string DISCONNECT = 'disconnect';
const string MESSAGE = 'message';
const string RECEIVE = 'Receive';
const string PACKET = 'Packet';
const string REQUEST = 'request';
const string CLOSE = 'Close';
const string TASK = 'Task';
const string FINISH = 'Finish';
const string PIPE_MESSAGE = 'PipeMessage';
const string WORKER_ERROR = 'WorkerError';
const string MANAGER_START = 'ManagerStart';
const string MANAGER_STOP = 'ManagerStop';
const string BEFORE_RELOAD = 'BeforeReload';
const string AFTER_RELOAD = 'AfterReload';
const string SERVER_TYPE_HTTP = 'http';
const string SERVER_TYPE_WEBSOCKET = 'ws';
const string SERVER_TYPE_TCP = 'tcp';
const string SERVER_TYPE_UDP = 'udp';
const string SERVER_TYPE_BASE = 'base';
const string STATUS_404_MESSAGE = '<h2>HTTP 404 Not Found</h2><hr><i>Powered by Swoole</i>';
const string STATUS_405_MESSAGE = '<h2>HTTP 405 Method allow</h2><hr><i>Powered by Swoole</i>';
const string OPTION_REACTOR_NUM = 'reactor_num';
const string OPTION_WORKER_NUM = 'worker_num';
const string OPTION_MAX_REQUEST = 'max_request';
const string OPTION_MAX_CONN = 'max_connection';
const string OPTION_TASK_WORKER_NUM = 'task_worker_num';
const string OPTION_TASK_IPC_MODE = 'task_ipc_mode';
const string OPTION_TASK_MAX_REQUEST = 'task_max_request';
const string OPTION_TASK_TMPDIR = 'task_tmpdir';
const string OPTION_TASK_ENABLE_COROUTINE = 'task_enable_coroutine';
const string OPTION_TASK_USE_OBJECT = 'task_use_object';
const string OPTION_DISPATCH_MODE = 'dispatch_mode';
const string OPTION_DISPATCH_FUNC = 'dispatch_func';
const string OPTION_MESSAGE_QUEUE_KEY = 'message_queue_key';
const string OPTION_DAEMONIZE = 'daemonize';
const string OPTION_BACKLOG = 'backlog';
const string OPTION_LOG_FILE = 'log_file';
const string OPTION_LOG_LEVEL = 'log_level';
const string OPTION_LOG_DATE_WITH_MICROSECONDS = 'log_date_with_microseconds';
const string OPTION_LOG_ROTATION = 'log_rotation';
const string OPTION_LOG_DATE_FORMAT = 'log_date_format';
const string OPTION_OPEN_TCP_KEEPALIVE = 'open_tcp_keepalive';
const string OPTION_HEARTBEAT_CHECK_INTERVAL = 'heartbeat_check_interval';
const string OPTION_HEARTBEAT_IDLE_TIME = 'heartbeat_idle_time';
const string OPTION_OPEN_EOF_CHECK = 'open_eof_check';
const string OPTION_OPEN_EOF_SPLIT = 'open_eof_split';
const string OPTION_PACKAGE_EOF = 'package_eof';
const string OPTION_OPEN_LENGTH_CHECK = 'open_length_check';
const string OPTION_PACKAGE_LENGTH_TYPE = 'package_length_type';
const string OPTION_PACKAGE_LENGTH_FUNC = 'package_length_func';
const string OPTION_PACKAGE_MAX_LENGTH = 'package_max_length';
const string OPTION_OPEN_HTTP_PROTOCOL = 'open_http_protocol';
const string OPTION_OPEN_MQTT_PROTOCOL = 'open_mqtt_protocol';
const string OPTION_OPEN_REDIS_PROTOCOL = 'open_redis_protocol';
const string OPTION_OPEN_WEBSOCKET_PROTOCOL = 'open_websocket_protocol';
const string OPTION_OPEN_WEBSOCKET_CLOSE_FRAME = 'open_websocket_close_frame';
const string OPTION_OPEN_TCP_NODELAY = 'open_tcp_nodelay';
const string OPTION_OPEN_CPU_AFFINITY = 'open_cpu_affinity';
const string OPTION_CPU_AFFINITY_IGNORE = 'cpu_affinity_ignore';
const string OPTION_TCP_DEFER_ACCEPT = 'tcp_defer_accept';
const string OPTION_SSL_CERT_FILE = 'ssl_cert_file';
const string OPTION_SSL_KEY_FILE = 'ssl_key_file';
const string OPTION_SSL_METHOD = 'ssl_method';
const string OPTION_SSL_PROTOCOLS = 'ssl_protocols';
const string OPTION_SSL_SNI_CERTS = 'ssl_sni_certs';
const string OPTION_SSL_CIPHERS = 'ssl_ciphers';
const string OPTION_SSL_VERIFY_PEER = 'ssl_verify_peer';
const string OPTION_SSL_ALLOW_SELF_SIGNED = 'ssl_allow_self_signed';
const string OPTION_SSL_CLIENT_CERT_FILE = 'ssl_client_cert_file';
const string OPTION_SSL_COMPRESS = 'ssl_compress';
const string OPTION_SSL_VERIFY_DEPTH = 'ssl_verify_depth';
const string OPTION_SSL_PREFER_SERVER_CIPHERS = 'ssl_prefer_server_ciphers';
const string OPTION_SSL_DHPARAM = 'ssl_dhparam';
const string OPTION_SSL_ECDH_CURVE = 'ssl_ecdh_curve';
const string OPTION_USER = 'user';
const string OPTION_GROUP = 'group';
const string OPTION_CHROOT = 'chroot';
const string OPTION_PID_FILE = 'pid_file';
const string OPTION_BUFFER_INPUT_SIZE = 'buffer_input_size';
const string OPTION_BUFFER_OUTPUT_SIZE = 'buffer_output_size';
const string OPTION_SOCKET_BUFFER_SIZE = 'socket_buffer_size';
const string OPTION_ENABLE_UNSAFE_EVENT = 'enable_unsafe_event';
const string OPTION_DISCARD_TIMEOUT_REQUEST = 'discard_timeout_request';
const string OPTION_ENABLE_REUSE_PORT = 'enable_reuse_port';
const string OPTION_ENABLE_DELAY_RECEIVE = 'enable_delay_receive';
const string OPTION_RELOAD_ASYNC = 'reload_async';
const string OPTION_MAX_WAIT_TIME = 'max_wait_time';
const string OPTION_TCP_FASTOPEN = 'tcp_fastopen';
const string OPTION_REQUEST_SLOWLOG_FILE = 'request_slowlog_file';
const string OPTION_ENABLE_COROUTINE = 'enable_coroutine';
const string OPTION_MAX_COROUTINE = 'max_coroutine';
const string OPTION_SEND_YIELD = 'send_yield';
const string OPTION_SEND_TIMEOUT = 'send_timeout';
const string OPTION_HOOK_FLAGS = 'hook_flags';
const string OPTION_BUFFER_HIGH_WATERMARK = 'buffer_high_watermark';
const string OPTION_BUFFER_LOW_WATERMARK = 'buffer_low_watermark';
const string OPTION_TCP_USER_TIMEOUT = 'tcp_user_timeout';
const string OPTION_STATS_FILE = 'stats_file';
const string OPTION_EVENT_OBJECT = 'event_object';
const string OPTION_START_SESSION_ID = 'start_session_id';
const string OPTION_SINGLE_THREAD = 'single_thread';
const string OPTION_MAX_QUEUED_BYTES = 'max_queued_bytes';
}
+8 -8
View File
@@ -1,8 +1,8 @@
<?php
namespace Server\Contract;
interface OnBeforeShutdown
{
}
<?php
namespace Kiri\Server\Contract;
interface OnBeforeShutdown
{
}
+22 -22
View File
@@ -1,22 +1,22 @@
<?php
namespace Server\Contract;
use Swoole\Server;
/**
*
*/
interface OnCloseInterface
{
/**
* @param Server $server
* @param int $fd
*/
public function onClose(Server $server, int $fd): void;
}
<?php
namespace Kiri\Server\Contract;
use Swoole\Server;
/**
*
*/
interface OnCloseInterface
{
/**
* @param Server $server
* @param int $fd
* @return void
*/
public function onClose(Server $server, int $fd): void;
}
+18 -18
View File
@@ -1,18 +1,18 @@
<?php
namespace Server\Contract;
use Swoole\Server;
interface OnConnectInterface
{
/**
* @param Server $server
* @param int $fd
* @return void
*/
public function onConnect(Server $server, int $fd): void;
}
<?php
namespace Kiri\Server\Contract;
use Swoole\Server;
interface OnConnectInterface
{
/**
* @param Server $server
* @param int $fd
* @return void
*/
public function onConnect(Server $server, int $fd): void;
}
+8 -9
View File
@@ -1,19 +1,18 @@
<?php
namespace Server\Contract;
namespace Kiri\Server\Contract;
use Swoole\Server;
use Swoole\WebSocket\Server;
interface OnDisconnectInterface
{
/**
* @param Server $server
* @param int $fd
*/
public function onDisconnect(Server $server, int $fd): void;
/**
* @param Server $server
* @param int $fd
* @return void
*/
public function onDisconnect(Server $server, int $fd): void;
}
+2 -5
View File
@@ -1,14 +1,10 @@
<?php
namespace Server\Contract;
namespace Kiri\Server\Contract;
use Swoole\Http\Request;
use Swoole\Http\Response;
/**
*
*/
interface OnHandshakeInterface
{
@@ -16,6 +12,7 @@ interface OnHandshakeInterface
/**
* @param Request $request
* @param Response $response
* @return void
*/
public function onHandshake(Request $request, Response $response): void;
+1 -1
View File
@@ -1,6 +1,6 @@
<?php
namespace Server\Contract;
namespace Kiri\Server\Contract;
use Swoole\Server;
use Swoole\WebSocket\Frame;
+8 -7
View File
@@ -1,18 +1,19 @@
<?php
namespace Server\Contract;
namespace Kiri\Server\Contract;
use Swoole\Http\Request;
use Swoole\WebSocket\Server;
use Swoole\Http\Server;
interface OnOpenInterface
{
/**
* @param Server $server
* @param Request $request
*/
public function onOpen(Server $server, Request $request): void;
/**
* @param Server $server
* @param Request $request
* @return void
*/
public function onOpen(Server $server, Request $request): void;
}
+19 -19
View File
@@ -1,19 +1,19 @@
<?php
namespace Server\Contract;
use Server\Abstracts\Server;
interface OnPacketInterface
{
/**
* @param Server $server
* @param string $data
* @param array $clientInfo
* @return mixed
*/
public function onPacket(Server $server, string $data, array $clientInfo): void;
}
<?php
namespace Kiri\Server\Contract;
use Swoole\Server;
interface OnPacketInterface
{
/**
* @param Server $server
* @param string $data
* @param array $clientInfo
* @return void
*/
public function onPacket(Server $server, string $data, array $clientInfo): void;
}
+21 -21
View File
@@ -1,21 +1,21 @@
<?php
namespace Server\Contract;
/**
* Interface OnPipeMessageInterface
* @package Server\Contract
*/
interface OnPipeMessageInterface
{
/**
*
*/
public function process(): void;
}
<?php
namespace Kiri\Server\Contract;
/**
* Interface OnPipeMessageInterface
* @package Server\Contract
*/
interface OnPipeMessageInterface
{
/**
*
*/
public function process(): void;
}
+25 -25
View File
@@ -1,25 +1,25 @@
<?php
namespace Server\Contract;
use Swoole\Process;
/**
* Interface BaseProcess
* @package Contract
*/
interface OnProcessInterface
{
/**
* @param Process $process
*/
public function process(Process $process): void;
}
<?php
namespace Kiri\Server\Contract;
use Swoole\Process;
/**
* Interface BaseProcess
* @package Contract
*/
interface OnProcessInterface
{
/**
* @param ?Process $process
*/
public function process(?Process $process): void;
}
+25 -25
View File
@@ -1,25 +1,25 @@
<?php
namespace Server\Contract;
use Swoole\Server;
/**
*
*/
interface OnReceiveInterface
{
/**
* @param Server $server
* @param int $fd
* @param int $reactor_id
* @param string $data
* @return void
*/
public function onReceive(Server $server, int $fd, int $reactor_id, string $data): void;
}
<?php
namespace Kiri\Server\Contract;
use Swoole\Server;
/**
*
*/
interface OnReceiveInterface
{
/**
* @param Server $server
* @param int $fd
* @param int $reactor_id
* @param string $data
* @return bool
*/
public function onReceive(Server $server, int $fd, int $reactor_id, string $data): bool;
}
-17
View File
@@ -1,17 +0,0 @@
<?php
namespace Server\Contract;
use Swoole\Server;
interface OnTaskInterface
{
public function execute();
public function finish(Server $server, int $task_id);
}
+18 -18
View File
@@ -1,18 +1,18 @@
<?php
namespace Server\Events;
use Swoole\Server;
class OnAfterReload
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
class OnAfterReload
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
+8
View File
@@ -0,0 +1,8 @@
<?php
namespace Kiri\Server\Events;
class OnAfterRequest
{
}
+19 -8
View File
@@ -1,8 +1,19 @@
<?php
namespace Server\Events;
class OnAfterWorkerStart
{
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
class OnAfterWorkerStart
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
}
+18 -18
View File
@@ -1,18 +1,18 @@
<?php
namespace Server\Events;
use Swoole\Server;
class OnBeforeReload
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
class OnBeforeReload
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
+18 -18
View File
@@ -1,18 +1,18 @@
<?php
namespace Server\Events;
use Swoole\Server;
class OnBeforeShutdown
{
/**
* @param Server|null $server
*/
public function __construct(public ?Server $server = null)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
class OnBeforeShutdown
{
/**
* @param Server|null $server
*/
public function __construct(public ?Server $server = null)
{
}
}
+18 -12
View File
@@ -1,12 +1,18 @@
<?php
namespace Server\Events;
class OnBeforeWorkerStart
{
public function __construct(public int $workerId)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
class OnBeforeWorkerStart
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server,public int $workerId)
{
}
}
+16 -16
View File
@@ -1,16 +1,16 @@
<?php
namespace Server\Events;
use Swoole\Server;
class OnManagerStart
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
class OnManagerStart
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
+18 -18
View File
@@ -1,18 +1,18 @@
<?php
namespace Server\Events;
use Swoole\Server;
class OnManagerStop
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
class OnManagerStop
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
+8
View File
@@ -0,0 +1,8 @@
<?php
namespace Kiri\Server\Events;
class OnProcessStart
{
}
+17
View File
@@ -0,0 +1,17 @@
<?php
namespace Kiri\Server\Events;
use Swoole\Process;
class OnProcessStop
{
/**
* @param Process $process
*/
public function __construct(Process $process)
{
}
}
+8 -8
View File
@@ -1,8 +1,8 @@
<?php
namespace Server\Events;
class OnServerBeforeStart
{
}
<?php
namespace Kiri\Server\Events;
class OnServerBeforeStart
{
}
+20 -18
View File
@@ -1,18 +1,20 @@
<?php
namespace Server\Events;
use Swoole\Server;
class OnShutdown
{
/**
* @param Server|null $server
*/
public function __construct(public ?Server $server = null)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Coroutine\Http\Server as CHServer;
use Swoole\Coroutine\Server as CServer;
use Swoole\Server;
class OnShutdown
{
/**
* @param Server|CHServer|CServer|null $server
*/
public function __construct(public Server|null|CHServer|CServer $server = null)
{
}
}
+18 -18
View File
@@ -1,18 +1,18 @@
<?php
namespace Server\Events;
use Swoole\Server;
class OnStart
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
class OnStart
{
/**
* @param Server $server
*/
public function __construct(public Server $server)
{
}
}
+25 -23
View File
@@ -1,23 +1,25 @@
<?php
namespace Server\Events;
use Swoole\Server;
/**
*
*/
class OnTaskerStart
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
}
<?php
namespace Kiri\Server\Events;
use Kiri\Exception\ConfigException;
use Swoole\Server;
use Kiri;
/**
*
*/
class OnTaskerStart
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
}
+26 -26
View File
@@ -1,26 +1,26 @@
<?php
namespace Server\Events;
use Swoole\Server;
/**
*
*/
class OnWorkerError
{
/**
* @param Server $server
* @param int $worker_id
* @param int $worker_pid
* @param int $exit_code
* @param int $signal
*/
public function __construct(public Server $server, public int $worker_id, public int $worker_pid, public int $exit_code, public int $signal)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
/**
*
*/
class OnWorkerError
{
/**
* @param Server $server
* @param int $worker_id
* @param int $worker_pid
* @param int $exit_code
* @param int $signal
*/
public function __construct(public Server $server, public int $worker_id, public int $worker_pid, public int $exit_code, public int $signal)
{
}
}
+23 -23
View File
@@ -1,23 +1,23 @@
<?php
namespace Server\Events;
use Swoole\Server;
/**
*
*/
class OnWorkerExit
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
/**
*
*/
class OnWorkerExit
{
/**
* @param Server|null $server
* @param int $workerId
*/
public function __construct(public ?Server $server, public int $workerId)
{
}
}
+25 -23
View File
@@ -1,23 +1,25 @@
<?php
namespace Server\Events;
use Swoole\Server;
/**
*
*/
class OnWorkerStart
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Server;
use Swoole\Coroutine\Server as CServer;
use Swoole\Coroutine\Http\Server as CHServer;
/**
*
*/
class OnWorkerStart
{
/**
* @param Server|CHServer|CServer|null $server
* @param int $workerId
*/
public function __construct(public Server|null|CHServer|CServer $server, public int $workerId)
{
}
}
+25 -23
View File
@@ -1,23 +1,25 @@
<?php
namespace Server\Events;
use Swoole\Server;
/**
*
*/
class OnWorkerStop
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
}
<?php
namespace Kiri\Server\Events;
use Swoole\Coroutine\Http\Server as CHServer;
use Swoole\Coroutine\Server as CServer;
use Swoole\Server;
/**
*
*/
class OnWorkerStop
{
/**
* @param Server|CHServer|CServer|null $server
* @param int $workerId
*/
public function __construct(public Server|null|CHServer|CServer $server, public int $workerId)
{
}
}
+34 -38
View File
@@ -1,38 +1,34 @@
<?php
namespace Server\Handler;
use Annotation\Inject;
use Server\Abstracts\Server;
use Exception;
use Server\Contract\OnPipeMessageInterface;
use Kiri\Events\EventDispatch;
/**
*
*/
class OnPipeMessage extends Server
{
/** @var EventDispatch */
#[Inject(EventDispatch::class)]
public EventDispatch $eventDispatch;
/**
* @param \Swoole\Server $server
* @param int $src_worker_id
* @param mixed $message
* @throws Exception
*/
public function onPipeMessage(\Swoole\Server $server, int $src_worker_id, mixed $message)
{
if (!is_object($message) || !($message instanceof OnPipeMessageInterface)) {
return;
}
call_user_func([$message, 'process'], $server, $src_worker_id);
}
}
<?php
namespace Kiri\Server\Handler;
use Exception;
use Kiri\Server\Abstracts\Server;
use Kiri\Server\Contract\OnPipeMessageInterface;
/**
*
*/
class OnPipeMessage extends Server
{
/**
* @param \Swoole\Server $server
* @param int $src_worker_id
* @param mixed $message
* @throws
*/
public function onPipeMessage(\Swoole\Server $server, int $src_worker_id, mixed $message): void
{
if (is_string($message)) {
$message = unserialize($message);
}
if (!is_object($message) || !($message instanceof OnPipeMessageInterface)) {
return;
}
call_user_func([$message, 'process'], $server, $src_worker_id);
}
}
+113
View File
@@ -0,0 +1,113 @@
<?php
declare(strict_types=1);
namespace Kiri\Server\Handler;
use Exception;
use Kiri\Di\Inject\Container;
use Kiri\Di\Context;
use Kiri\Di\Interface\ResponseEmitterInterface;
use Kiri\Router\Base\ExceptionHandlerDispatcher;
use Kiri\Router\Constrict\ConstrictRequest as CQ;
use Kiri\Router\Constrict\ConstrictResponse;
use Kiri\Router\DataGrip;
use Kiri\Router\Interface\ExceptionHandlerInterface;
use Kiri\Router\Interface\OnRequestInterface;
use Kiri\Router\RouterCollector;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\ContainerInterface;
use Psr\Container\NotFoundExceptionInterface;
use Psr\Http\Message\RequestInterface;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use ReflectionException;
use Swoole\Http\Request;
use Swoole\Http\Response;
use Throwable;
/**
* OnRequest event
*/
class OnRequest implements OnRequestInterface
{
/**
* @var RouterCollector
*/
public RouterCollector $router;
/**
* @var ExceptionHandlerInterface
*/
public ExceptionHandlerInterface $exception;
/**
* @var ResponseEmitterInterface
*/
public ResponseEmitterInterface $responseEmitter;
/**
* @var ConstrictResponse
*/
#[Container(ConstrictResponse::class)]
public ConstrictResponse $constrictResponse;
/**
* @param ResponseInterface $response
* @param ContainerInterface $container
* @param DataGrip $dataGrip
* @throws
*/
public function __construct(public ResponseInterface $response, public ContainerInterface $container,
public DataGrip $dataGrip)
{
$this->responseEmitter = $this->response->emmit;
$exception = \config('exception.http');
if (!in_array(ExceptionHandlerInterface::class, class_implements($exception))) {
$exception = ExceptionHandlerDispatcher::class;
}
$this->exception = $this->container->get($exception);
$this->router = $this->dataGrip->get(ROUTER_TYPE_HTTP);
}
/**
* @param Request $request
* @param Response $response
* @throws
*/
public function onRequest(Request $request, Response $response): void
{
/** @var CQ $PsrRequest */
try {
$PsrRequest = $this->initRequestAndResponse($request);
$PsrResponse = $this->router->query($request->server['path_info'], $request->getMethod())
->run($PsrRequest);
} catch (Throwable $throwable) {
$PsrResponse = $this->exception->emit($throwable, $this->constrictResponse);
} finally {
$this->responseEmitter->response($PsrResponse, $response, $PsrRequest);
}
}
/**
* @param Request $request
* @return ServerRequestInterface
*/
public function initRequestAndResponse(Request $request): ServerRequestInterface
{
$response = new ConstrictResponse($this->response->contentType);
Context::set(ResponseInterface::class, $response);
return Context::set(RequestInterface::class, CQ::builder($request));
}
}
+66 -58
View File
@@ -1,58 +1,66 @@
<?php
namespace Server\Handler;
use Annotation\Inject;
use Kiri\Events\EventDispatch;
use Kiri\Exception\ConfigException;
use Server\Abstracts\Server;
use Server\Events\OnBeforeShutdown;
use Server\Events\OnShutdown;
use Server\Events\OnStart;
/**
* Class OnServerDefault
* @package Server\Manager
*/
class OnServer extends Server
{
/**
* @var EventDispatch
*/
#[Inject(EventDispatch::class)]
public EventDispatch $eventDispatch;
/**
* @param \Swoole\Server $server
* @throws ConfigException
*/
public function onStart(\Swoole\Server $server)
{
$this->setProcessName(sprintf('start[%d].server', $server->master_pid));
$this->eventDispatch->dispatch(new OnStart($server));
}
/**
* @param \Swoole\Server $server
*/
public function onBeforeShutdown(\Swoole\Server $server)
{
$this->eventDispatch->dispatch(new OnBeforeShutdown($server));
}
/**
* @param \Swoole\Server $server
*/
public function onShutdown(\Swoole\Server $server)
{
$this->eventDispatch->dispatch(new OnShutdown($server));
}
}
<?php
namespace Kiri\Server\Handler;
use Kiri;
use Kiri\Di\Inject\Container;
use Kiri\Error\StdoutLogger;
use Psr\Log\LoggerInterface;
use ReflectionException;
use Kiri\Server\Abstracts\Server;
use Kiri\Server\Events\OnBeforeShutdown;
use Kiri\Server\Events\OnShutdown;
use Kiri\Server\Events\OnStart;
use Swoole\Server as SServer;
/**
* Class OnServerDefault
* @package Server\Manager
*/
class OnServer extends Server
{
/**
* @var StdoutLogger
*/
#[Container(LoggerInterface::class)]
public StdoutLogger $logger;
/**
* @param SServer $server
* @throws
*/
public function onStart(SServer $server): void
{
Kiri::setProcessName(sprintf('start[%d].server', $server->master_pid));
foreach (config('server.ports') as $value) {
$this->logger->alert('Listen ' . $value['type'] . ' address ' . $value['host'] . '::' . $value['port']);
}
event(new OnStart($server));
}
/**
* @param SServer $server
* @throws
*/
public function onBeforeShutdown(SServer $server): void
{
event(new OnBeforeShutdown($server));
}
/**
* @param SServer $server
* @throws
*/
public function onShutdown(SServer $server): void
{
@unlink(storage('.swoole.pid'));
event(new OnShutdown($server));
}
}
+43 -48
View File
@@ -1,48 +1,43 @@
<?php
namespace Server\Handler;
use Annotation\Inject;
use Kiri\Events\EventDispatch;
use Server\Abstracts\Server;
use Kiri\Exception\ConfigException;
use Server\Events\OnManagerStart;
use Server\Events\OnManagerStop;
/**
* Class OnServerManager
* @package Server\Manager
*/
class OnServerManager extends Server
{
/**
* @var EventDispatch
*/
#[Inject(EventDispatch::class)]
public EventDispatch $eventDispatch;
/**
* @param \Swoole\Server $server
* @throws ConfigException
*/
public function onManagerStart(\Swoole\Server $server)
{
$this->setProcessName(sprintf('manger[%d].0', $server->manager_pid));
$this->eventDispatch->dispatch(new OnManagerStart($server));
}
/**
* @param \Swoole\Server $server
*/
public function onManagerStop(\Swoole\Server $server)
{
$this->eventDispatch->dispatch(new OnManagerStop($server));
}
}
<?php
namespace Kiri\Server\Handler;
use Kiri;
use ReflectionException;
use Kiri\Server\Abstracts\Server;
use Kiri\Server\Events\OnManagerStart;
use Kiri\Server\Events\OnManagerStop;
/**
* Class OnServerManager
* @package Server\Manager
*/
class OnServerManager extends Server
{
/**
* @param \Swoole\Server $server
* @throws
*/
public function onManagerStart(\Swoole\Server $server): void
{
Kiri::setProcessName(sprintf('manger process[%d]', $server->manager_pid));
event(new OnManagerStart($server));
}
/**
* @param \Swoole\Server $server
* @return void
* @throws
*/
public function onManagerStop(\Swoole\Server $server): void
{
event(new OnManagerStop($server));
}
}
-43
View File
@@ -1,43 +0,0 @@
<?php
namespace Server\Handler;
use Annotation\Inject;
use Kiri\Events\EventDispatch;
use Server\Events\OnAfterReload;
use Server\Events\OnBeforeReload;
use Swoole\Server;
/**
*
*/
class OnServerReload
{
/**
* @var EventDispatch
*/
#[Inject(EventDispatch::class)]
public EventDispatch $eventDispatch;
/**
* @param Server $server
*/
public function onBeforeReload(Server $server)
{
$this->eventDispatch->dispatch(new OnBeforeReload($server));
}
/**
* @param Server $server
*/
public function onAfterReload(Server $server)
{
$this->eventDispatch->dispatch(new OnAfterReload($server));
}
}
-101
View File
@@ -1,101 +0,0 @@
<?php
namespace Server\Handler;
use Annotation\Inject;
use Kiri\Abstracts\Logger;
use Kiri\Exception\ConfigException;
use Kiri\Kiri;
use ReflectionException;
use Server\Contract\OnTaskInterface;
use Swoole\Server;
/**
* Class OnServerTask
* @package Server\Task
*/
class OnServerTask
{
#[Inject(Logger::class)]
public Logger $logger;
/**
* @param Server $server
* @param int $task_id
* @param int $src_worker_id
* @param mixed $data
* @throws ConfigException
*/
public function onTask(Server $server, int $task_id, int $src_worker_id, mixed $data)
{
try {
$data = $this->resolve($data);
} catch (\Throwable $exception) {
$data = jTraceEx($exception);
$this->logger->error('task', [$data]);
} finally {
$server->finish($data);
}
}
/**
* @param Server $server
* @param Server\Task $task
* @throws ConfigException
*/
public function onCoroutineTask(Server $server, Server\Task $task)
{
try {
$data = $this->resolve($task->data);
} catch (\Throwable $exception) {
$data = jTraceEx($exception);
$this->logger->error('task', [$data]);
} finally {
$server->finish($data);
}
}
/**
* @param $data
* @return null
* @throws ReflectionException
*/
private function resolve($data)
{
[$class, $params] = json_encode($data, true);
$reflect = Kiri::getDi()->getReflect($class);
if (!$reflect->isInstantiable()) {
return null;
}
$class = $reflect->newInstanceArgs($params);
return $class->execute();
}
/**
* @param Server $server
* @param int $task_id
* @param mixed $data
*/
public function onFinish(Server $server, int $task_id, mixed $data)
{
if (!($data instanceof OnTaskInterface)) {
return;
}
$data->finish($server, $task_id);
}
}
+152 -121
View File
@@ -1,121 +1,152 @@
<?php
namespace Server\Handler;
use Annotation\Inject;
use Exception;
use Kiri\Abstracts\Config;
use Kiri\Core\Help;
use Kiri\Events\EventDispatch;
use Kiri\Kiri;
use Server\Events\OnAfterWorkerStart;
use Server\Events\OnBeforeWorkerStart;
use Server\Events\OnTaskerStart as OnTaskStart;
use Server\Events\OnWorkerError;
use Server\Events\OnWorkerExit;
use Server\Events\OnWorkerStart;
use Server\Events\OnWorkerStop;
use Swoole\Server;
use Swoole\Timer;
/**
* Class OnServerWorker
* @package Server\Worker
*/
class OnServerWorker extends \Server\Abstracts\Server
{
/**
* @var EventDispatch
*/
#[Inject(EventDispatch::class)]
public EventDispatch $eventDispatch;
/**
* @param Server $server
* @param int $workerId
* @throws Exception
*/
public function onWorkerStart(Server $server, int $workerId)
{
$this->eventDispatch->dispatch(new OnBeforeWorkerStart($workerId));
if ($workerId < $server->setting['worker_num']) {
$this->eventDispatch->dispatch(new OnWorkerStart($server, $workerId));
$this->setProcessName(sprintf('Worker[%d].%d', $server->worker_pid, $workerId));
set_env('environmental', Kiri::WORKER);
} else {
$this->eventDispatch->dispatch(new OnTaskStart($server, $workerId));
$this->setProcessName(sprintf('Tasker[%d].%d', $server->worker_pid, $workerId));
set_env('environmental', Kiri::TASK);
}
$this->eventDispatch->dispatch(new OnAfterWorkerStart());
}
/**
* @param Server $server
* @param int $workerId
* @throws Exception
*/
public function onWorkerStop(Server $server, int $workerId)
{
Timer::clearAll();
$this->eventDispatch->dispatch(new OnWorkerStop($server, $workerId));
}
/**
* @param Server $server
* @param int $workerId
* @throws Exception
*/
public function onWorkerExit(Server $server, int $workerId)
{
set_env('state', 'exit');
$this->eventDispatch->dispatch(new OnWorkerExit($server, $workerId));
}
/**
* @param Server $server
* @param int $worker_id
* @param int $worker_pid
* @param int $exit_code
* @param int $signal
* @throws Exception
*/
public function onWorkerError(Server $server, int $worker_id, int $worker_pid, int $exit_code, int $signal)
{
$this->eventDispatch->dispatch(new OnWorkerError($server, $worker_id, $worker_pid, $exit_code, $signal));
$message = sprintf('Worker#%d::%d error stop. signal %d, exit_code %d, msg %s',
$worker_id, $worker_pid, $signal, $exit_code, swoole_strerror(swoole_last_error(), 9)
);
$this->logger->error($message);
// $this->system_mail($message);
}
/**
* @param $messageContent
* @throws Exception
*/
protected function system_mail($messageContent)
{
try {
$email = Config::get('email');
if (!empty($email) && ($email['enable'] ?? false) == true) {
Help::sendEmail($email, 'Service Error', $messageContent);
}
} catch (\Throwable $e) {
error($e, 'email');
}
}
}
<?php
namespace Kiri\Server\Handler;
use Kiri;
use Kiri\Core\Help;
use Kiri\Events\EventDispatch;
use Kiri\Server\Events\OnAfterWorkerStart;
use Kiri\Server\Events\OnBeforeWorkerStart;
use Kiri\Server\Events\OnTaskerStart as OnTaskerStart;
use Kiri\Server\Events\OnWorkerError;
use Kiri\Server\Events\OnWorkerExit;
use Kiri\Server\Events\OnWorkerStart;
use Kiri\Server\Events\OnWorkerStop;
use Swoole\Server;
use Kiri\Di\Inject\Container;
use Swoole\Timer;
use Throwable;
use function config;
/**
* Class OnServerWorker
* @package Server\Worker
*/
class OnServerWorker extends Kiri\Server\Abstracts\Server
{
/**
* @var EventDispatch
*/
#[Container(EventDispatch::class)]
public EventDispatch $dispatch;
/**
* @return void
*/
public function init(): void
{
on(OnBeforeWorkerStart::class, [$this, 'onWorkerNameAlias']);
}
/**
* @param OnBeforeWorkerStart $workerStart
* @return void
*/
public function onWorkerNameAlias(OnBeforeWorkerStart $workerStart): void
{
set_env('environmental_workerId', $workerStart->workerId);
if ($workerStart->workerId < $workerStart->server->setting['worker_num']) {
$this->processName($workerStart->server, 'Worker');
set_env('environmental', Kiri::WORKER);
} else {
$this->processName($workerStart->server, 'Tasker');
set_env('environmental', Kiri::TASK);
}
}
/**
* @param Server $server
* @param int $workerId
* @return void
* @throws
*/
public function onWorkerStart(Server $server, int $workerId): void
{
$this->dispatch->dispatch(new OnBeforeWorkerStart($server, $workerId));
if ($workerId < $server->setting['worker_num']) {
$this->dispatch->dispatch(new OnWorkerStart($server, $workerId));
} else {
$this->dispatch->dispatch(new OnTaskerStart($server, $workerId));
}
$this->dispatch->dispatch(new OnAfterWorkerStart($server, $workerId));
}
/**
* @param Server $server
* @param string $prefix
* @return void
*/
protected function processName(Server $server, string $prefix): void
{
Kiri::setProcessName(sprintf($prefix . ' Process[%d].%d', $server->worker_pid, $server->worker_id));
}
/**
* @param Server $server
* @param int $workerId
* @throws
*/
public function onWorkerStop(Server $server, int $workerId): void
{
event(new OnWorkerStop($server, $workerId));
Timer::clearAll();
}
/**
* @param Server $server
* @param int $workerId
* @throws
*/
public function onWorkerExit(Server $server, int $workerId): void
{
event(new OnWorkerExit($server, $workerId));
}
/**
* @param Server $server
* @param int $worker_id
* @param int $worker_pid
* @param int $exit_code
* @param int $signal
* @throws
*/
public function onWorkerError(Server $server, int $worker_id, int $worker_pid, int $exit_code, int $signal): void
{
event(new OnWorkerError($server, $worker_id, $worker_pid, $exit_code, $signal));
debug_print_backtrace(DEBUG_BACKTRACE_PROVIDE_OBJECT);
$message = sprintf('Worker#%d::%d error stop. signal %d, exit_code %d, msg %s', $worker_id, $worker_pid, $signal, $exit_code, swoole_strerror(swoole_last_error(), $signal));
error($message);
$this->system_mail($message);
}
/**
* @param $messageContent
* @throws
*/
protected function system_mail($messageContent): void
{
try {
$email = config('email', ['enable' => false]);
if (!empty($email) && ($email['enable'] ?? false)) {
Help::sendEmail($email, 'Service Error', $messageContent);
}
} catch (Throwable $e) {
error($e, ['email']);
}
}
}
+376
View File
@@ -0,0 +1,376 @@
<?php
declare(strict_types=1);
namespace Kiri\Server;
use Kiri\Di\Context;
use Kiri\Router\Router;
use Kiri\Server\Abstracts\BaseProcess;
use Kiri\Server\Events\OnWorkerStart;
use Swoole\Coroutine;
use Swoole\Event;
use Swoole\Process;
class HotReload extends BaseProcess
{
/**
* @var array|mixed
*/
private array $watchFiles = [];
private array $md5Map = [];
/**
* @var array|string[]
*/
private array $dirs = [APP_PATH . 'app', APP_PATH . 'routes'];
private bool $forceFile = false;
protected mixed $inotify = null;
/**
* @param Router $router
*/
public function __construct(public Router $router)
{
on(OnWorkerStart::class, [$this->router, 'scan_build_route']);
$this->forceFile = \config('reload.forceFile', false);
}
/**
* @return string
*/
public function getName(): string
{
// TODO: Change the autogenerated stub
return '[' . \config('id', 'system-service') . '].hot.load';
}
/**
* @return $this
*/
public function onSigterm(): static
{
// TODO: Implement onSigterm() method.
if (Context::inCoroutine()) {
Coroutine::create(fn() => $this->onShutdown(Coroutine::waitSignal(SIGTERM | SIGINT)));
} else {
\pcntl_signal(SIGTERM, [$this, 'onStop']);
}
return $this;
}
/**
* @param $data
* @return void
* @throws
*/
public function onStop($data): void
{
$this->clearWatch();
if (is_resource($this->inotify)) {
Event::del($this->inotify);
}
$this->onShutdown($data);
}
/**
* @param Process|null $process
* @return void
* @throws
*/
public function process(?Process $process): void
{
// TODO: Implement process() method.
if (!$this->forceFile && extension_loaded('inotify')) {
$this->onInotifyReload();
} else {
$this->onCrontabReload();
}
$process->exit(0);
}
/**
* @return void
* @throws
*/
private function onCrontabReload(): void
{
$this->loadDirs();
$this->tick();
}
/**
* @return void
* @throws
*/
private function onInotifyReload(): void
{
$this->inotify = inotify_init();
foreach ($this->dirs as $dir) {
if (!is_dir($dir)) {
continue;
}
$this->watch(rtrim($dir, '/'));
}
Event::add($this->inotify, fn() => $this->check());
Event::cycle(fn() => function () {
Event::dispatch();
}, true);
Event::wait();
}
/**
* @param bool $isReload
* @throws
*/
private function loadDirs(bool $isReload = false): void
{
foreach ($this->dirs as $value) {
if (is_bool($path = realpath($value))) {
continue;
}
if (!is_dir($path)) continue;
$this->loadByDir($path, $isReload);
}
}
/**
* @throws
*/
public function tick(): void
{
$isReloading = Context::get('isReloading', false);
if ($isReloading) {
return;
}
$pid = (int)file_get_contents(storage('.swoole.pid'));
if ($pid <= 0 || !Process::kill($pid, 0) || $this->isStop()) {
return;
}
$this->loadDirs(true);
sleep(2);
$this->tick();
}
/**
* @param $path
* @param bool $isReload
* @return void
* @throws
*/
private function loadByDir($path, bool $isReload = false): void
{
if (!is_string($path) || $this->isStop()) {
return;
}
$path = rtrim($path, '/');
foreach (glob(realpath($path) . '/*') as $value) {
if ($this->isStop()) {
break;
}
if (is_dir($value)) {
$this->loadByDir($value, $isReload);
}
if (is_file($value)) {
if ($this->checkFile($value, $isReload)) {
$this->timerReload();
break;
}
}
}
}
/**
* @param $value
* @param $isReload
* @return bool
*/
private function checkFile($value, $isReload): bool
{
$md5 = md5($value);
$mTime = filectime($value);
if (!isset($this->md5Map[$md5])) {
if ($isReload) {
return true;
}
$this->md5Map[$md5] = $mTime;
} else {
if ($this->md5Map[$md5] != $mTime) {
if ($isReload) {
return true;
}
$this->md5Map[$md5] = $mTime;
}
}
return false;
}
/**
* 开始监听
*/
public function check(): void
{
if (!($events = inotify_read($this->inotify))) {
return;
}
if (Context::exists('isReloading')) {
return;
}
$eventList = [IN_CREATE, IN_DELETE, IN_MODIFY, IN_MOVED_TO, IN_MOVED_FROM];
foreach ($events as $ev) {
if (empty($ev['name'])) {
continue;
}
if ($ev['mask'] == IN_IGNORED) {
continue;
}
if (!in_array($ev['mask'], $eventList)) {
continue;
}
$fileType = strstr($ev['name'], '.');
//非重启类型
if ($fileType !== '.php') {
continue;
}
if (Context::exists('swoole_timer_after')) {
return;
}
$int = @swoole_timer_after(2000, fn() => $this->reload());
Context::set('swoole_timer_after', $int);
Context::set('isReloading', true);
}
}
/**
* @throws
*/
public function reload(): void
{
$this->trigger_reload();
$this->clearWatch();
foreach ($this->dirs as $root) {
$this->watch($root);
}
Context::remove('swoole_timer_after');
Context::remove('isReloading');
$this->md5Map = [];
}
/**
* @throws
*/
public function timerReload(): void
{
Context::set('isReloading', true);
$this->trigger_reload();
Context::set('swoole_timer_after', -1);
$this->loadDirs();
Context::set('isReloading', false);
$this->tick();
}
/**
* 重启
* @throws
*/
public function trigger_reload(): void
{
$this->logger->failure('Wait trigger server Reload' . PHP_EOL);
di(ServerInterface::class)->reload(false);
}
/**
* @throws
*/
public function clearWatch(): void
{
foreach ($this->watchFiles as $wd) {
try {
inotify_rm_watch($this->inotify, $wd);
} catch (\Throwable $exception) {
trigger_print_error($exception, 'throwable');
}
}
$this->watchFiles = [];
}
/**
* @param $dir
* @return bool
* @throws
*/
public function watch($dir): bool
{
//目录不存在
if (!is_dir($dir)) {
return trigger_print_error("[$dir] is not a directory.");
}
//避免重复监听
if (isset($this->watchFiles[$dir])) {
return FALSE;
}
if (in_array($dir, [APP_PATH . 'commands', APP_PATH . '.git', APP_PATH . '.gitee'])) {
return FALSE;
}
$wd = @inotify_add_watch($this->inotify, $dir, IN_MODIFY | IN_DELETE | IN_CREATE | IN_MOVE);
$this->watchFiles[$dir] = $wd;
$files = scandir($dir);
foreach ($files as $f) {
if ($f == '.' or $f == '..' or $f == 'runtime' or preg_match('/\.txt/', $f) or preg_match('/\.sql/', $f) or preg_match('/\.log/', $f)) {
continue;
}
$path = $dir . '/' . $f;
//递归目录
if (is_dir($path)) {
$this->watch($path);
continue;
}
//检测文件类型
if (strstr($f, '.') == '.php') {
$wd = @inotify_add_watch($this->inotify, $path, IN_MODIFY | IN_DELETE | IN_CREATE | IN_MOVE);
$this->watchFiles[$path] = $wd;
}
}
return TRUE;
}
}
-135
View File
@@ -1,135 +0,0 @@
<?php
namespace Server;
use Annotation\Inject;
use Exception;
use Http\Handler\Abstracts\HttpService;
use JetBrains\PhpStorm\Pure;
use Kiri\Abstracts\Config;
use Kiri\Events\EventDispatch;
use Kiri\Exception\ConfigException;
use Server\Events\OnShutdown;
defined('PID_PATH') or define('PID_PATH', APP_PATH . 'storage/server.pid');
/**
* Class Server
* @package Http
*/
class Server extends HttpService
{
private array $process = [
];
/**
* @Inject ServerManager
* @var null|ServerManager
*/
#[Inject(ServerManager::class)]
public ?ServerManager $manager = null;
private mixed $daemon = 0;
/** @var EventDispatch */
#[Inject(EventDispatch::class)]
public EventDispatch $eventDispatch;
/**
*
*/
public function init()
{
}
/**
* @param $process
*/
public function addProcess($process)
{
$this->process[] = $process;
}
/**
* @return string start server
*
* start server
* @throws ConfigException
* @throws Exception
*/
public function start(): string
{
$this->manager->initBaseServer(Config::get('server', [], true), $this->daemon);
$rpcService = Config::get('rpc', []);
if (!empty($rpcService)) {
$this->manager->addListener($rpcService['type'], $rpcService['host'], $rpcService['port'],
$rpcService['mode'], $rpcService);
}
$processes = array_merge($this->process, Config::get('processes', []));
foreach ($processes as $process) {
$this->manager->addProcess($process);
}
return $this->manager->getServer()->start();
}
/**
* @return void
*
* start server
* @throws Exception
*/
public function shutdown()
{
$configs = Config::get('server', [], true);
foreach ($this->manager->sortService($configs['ports'] ?? []) as $config) {
$this->manager->stopServer($config['port']);
}
$this->eventDispatch->dispatch(new OnShutdown());
}
/**
* @return bool
* @throws ConfigException
*/
public function isRunner(): bool
{
return $this->manager->isRunner();
}
/**
* @param $daemon
* @return Server
*/
public function setDaemon($daemon): static
{
if (!in_array($daemon, [0, 1])) {
return $this;
}
$this->daemon = $daemon;
return $this;
}
/**
* @return \Swoole\Http\Server|\Swoole\Server|\Swoole\WebSocket\Server|null
*/
#[Pure] public function getServer(): \Swoole\Http\Server|\Swoole\Server|\Swoole\WebSocket\Server|null
{
return $this->manager->getServer();
}
}
+131 -116
View File
@@ -1,116 +1,131 @@
<?php
declare(strict_types=1);
namespace Server;
use Annotation\Inject;
use Exception;
use Kiri\Abstracts\Config;
use Kiri\Events\EventDispatch;
use Kiri\Exception\ConfigException;
use Kiri\Kiri;
use Server\Events\OnServerBeforeStart;
use Swoole\Coroutine;
use Symfony\Component\Console\Command\Command;
use Symfony\Component\Console\Input\InputArgument;
use Symfony\Component\Console\Input\InputInterface;
use Symfony\Component\Console\Input\InputOption;
use Symfony\Component\Console\Output\OutputInterface;
/**
* Class Command
* @package Http
*/
class ServerCommand extends Command
{
const ACTIONS = ['start', 'stop', 'restart'];
/**
* @var EventDispatch
*/
#[Inject(EventDispatch::class)]
public EventDispatch $eventProvider;
/**
*
*/
protected function configure()
{
$this->setName('sw:server')
->setDescription('server start|stop|reload|restart')
->addArgument('action', InputArgument::OPTIONAL, 'run action', 'start')
->addOption('daemon', 'd', InputOption::VALUE_OPTIONAL, 'is run daemonize');
}
/**
* @param InputInterface $input
* @param OutputInterface $output
* @return int
* @throws Exception
*/
public function execute(InputInterface $input, OutputInterface $output): int
{
$manager = Kiri::app()->getServer();
$manager->setDaemon((int)!is_null($input->getOption('daemon')));
if (is_null($input->getArgument('action'))) {
$input->setArgument('action', 'restart');
}
if (!in_array($input->getArgument('action'), self::ACTIONS)) {
throw new Exception('I don\'t know what I want to do.');
}
if ($manager->isRunner() && $input->getArgument('action') == 'start') {
throw new Exception('Service is running. Please use restart.');
}
$manager->shutdown();
if ($input->getArgument('action') == 'stop') {
return 0;
}
return $this->generate_runtime_builder($manager);
}
/**
* @throws ConfigException
*/
private function configure_set()
{
$enable_coroutine = Config::get('servers.settings.enable_coroutine', false);
if ($enable_coroutine != true) {
return;
}
Coroutine::set([
'hook_flags' => SWOOLE_HOOK_ALL ^ SWOOLE_HOOK_BLOCKING_FUNCTION,
'enable_deadlock_check' => FALSE,
'exit_condition' => function () {
return Coroutine::stats()['coroutine_num'] === 0;
}
]);
}
/**
* @param $manager
* @return int
* @throws ConfigException
* @throws Exception
*/
private function generate_runtime_builder($manager): int
{
$this->configure_set();
Kiri::app()->getRouter()->read_files();
$this->eventProvider->dispatch(new OnServerBeforeStart());
$manager->start();
return 1;
}
}
<?php
declare(strict_types=1);
namespace Kiri\Server;
use Exception;
use Kiri;
use Kiri\Events\EventDispatch;
use Kiri\Router\Router;
use Kiri\Server\Abstracts\AsyncServer;
use Kiri\Server\Events\OnShutdown;
use ReflectionException;
use Symfony\Component\Console\Command\Command;
use Symfony\Component\Console\Input\InputArgument;
use Symfony\Component\Console\Input\InputInterface;
use Symfony\Component\Console\Input\InputOption;
use Symfony\Component\Console\Output\OutputInterface;
use function config;
defined('ROUTER_TYPE_HTTP') or define('ROUTER_TYPE_HTTP', 'http');
defined('PID_PATH') or define('PID_PATH', APP_PATH . 'storage/server.pid');
/**
* Class Command
* @package Http
*/
class ServerCommand extends Command
{
public AsyncServer $manager;
public State $state;
public EventDispatch $dispatch;
public Router $router;
/**
* @param string|null $name
* @throws Exception
*/
public function __construct(string $name = null)
{
parent::__construct($name);
$container = Kiri::getDi();
$this->manager = $container->get(AsyncServer::class);
$this->state = $container->get(State::class);
$this->dispatch = $container->get(EventDispatch::class);
$this->router = $container->get(Router::class);
}
/**
* @return void
*/
protected function configure(): void
{
$this->setName('sw:server')
->setDescription('server start|stop|reload|restart')
->addArgument('action', InputArgument::OPTIONAL, 'run action', 'start')
->addOption('daemon', 'd', InputOption::VALUE_NONE, 'is run daemonize');
}
/**
* @param InputInterface $input
* @param OutputInterface $output
* @return int
* @throws
*/
public function execute(InputInterface $input, OutputInterface $output): int
{
return match ($input->getArgument('action')) {
'restart' => $this->restart($input),
'stop' => $this->stop(),
'start' => $this->start($input),
default =>
throw new Exception('I don\'t know what I want to do.')
};
}
/**
* @param InputInterface $input
* @return int
* @throws
*/
protected function restart(InputInterface $input): int
{
$this->stop();
$this->start($input);
return 1;
}
/**
* @return int
* @throws
*/
protected function stop(): int
{
$configs = config('server', []);
$instances = $this->manager->sortService($configs['ports'] ?? []);
foreach ($instances as $config) {
$this->state->exit($config->port);
}
$this->dispatch->dispatch(new OnShutdown());
return 1;
}
/**
* @param InputInterface $input
* @return int
* @throws
*/
protected function start(InputInterface $input): int
{
$daemon = (int)$input->getOption('daemon');
if (config('reload.hot', false) === true) {
$this->manager->addProcess(HotReload::class);
} else {
$this->router->scan_build_route();
}
$this->manager->addProcess(config('processes', []));
$this->manager->initCoreServers(config('server', []), $daemon);
$this->manager->start();
return 1;
}
}
+34
View File
@@ -0,0 +1,34 @@
<?php
namespace Kiri\Server;
use Swoole\Server;
/**
* @mixin Server
*/
interface ServerInterface
{
/**
* @param array $service
* @param int $daemon
* @return void
*/
public function initCoreServers(array $service, int $daemon = 0): void;
/**
* @param Config $config
* @return void
*/
public function addListener(Config $config): void;
/**
* @return void
*/
public function start(): void;
}
-442
View File
@@ -1,442 +0,0 @@
<?php
namespace Server;
use Annotation\Inject;
use Exception;
use Kiri\Abstracts\Config;
use Kiri\Di\ContainerInterface;
use Kiri\Error\Logger;
use Kiri\Exception\ConfigException;
use Kiri\Kiri;
use ReflectionException;
use Server\Abstracts\BaseProcess;
use Server\Contract\OnCloseInterface;
use Server\Contract\OnConnectInterface;
use Server\Contract\OnDisconnectInterface;
use Server\Contract\OnHandshakeInterface;
use Server\Contract\OnMessageInterface;
use Server\Contract\OnPacketInterface;
use Server\Contract\OnProcessInterface;
use Server\Contract\OnReceiveInterface;
use Server\Contract\OnTaskInterface;
use Server\Handler\OnPipeMessage;
use Server\Handler\OnServer;
use Server\Handler\OnServerManager;
use Server\Handler\OnServerReload;
use Server\Handler\OnServerTask;
use Server\Handler\OnServerWorker;
use Swoole\Http\Server as HServer;
use Swoole\Process;
use Swoole\Server;
use Swoole\Server\Port;
use Swoole\WebSocket\Server as WServer;
/**
* Class OnServerManager
* @package Http\Service
*/
class ServerManager
{
use TraitServer;
/** @var string */
public string $host = '';
public int $port = 0;
#[Inject(Logger::class)]
public Logger $logger;
/** @var array<string,Port> */
public array $ports = [];
public int $mode = SWOOLE_TCP;
private Server|null $server = null;
/**
* @var ContainerInterface
*/
#[Inject(ContainerInterface::class)]
public ContainerInterface $container;
const DEFAULT_EVENT = [
Constant::WORKER_START => [OnServerWorker::class, 'onWorkerStart'],
Constant::WORKER_EXIT => [OnServerWorker::class, 'onWorkerExit'],
Constant::WORKER_STOP => [OnServerWorker::class, 'onWorkerStop'],
Constant::WORKER_ERROR => [OnServerWorker::class, 'onWorkerError'],
Constant::MANAGER_START => [OnServerManager::class, 'onManagerStart'],
Constant::MANAGER_STOP => [OnServerManager::class, 'onManagerStop'],
Constant::BEFORE_RELOAD => [OnServerReload::class, 'onBeforeReload'],
Constant::AFTER_RELOAD => [OnServerReload::class, 'onAfterReload'],
Constant::START => [OnServer::class, 'onStart'],
Constant::BEFORE_SHUTDOWN => [OnServer::class, 'onBeforeShutdown'],
Constant::SHUTDOWN => [OnServer::class, 'onShutdown'],
];
private array $eventInterface = [
OnReceiveInterface::class => 'receive',
OnPacketInterface::class => 'packet',
OnHandshakeInterface::class => 'handshake',
OnMessageInterface::class => 'message',
OnConnectInterface::class => 'connect',
OnCloseInterface::class => 'close',
OnDisconnectInterface::class => 'disconnect'
];
/**
* @return Server|WServer|HServer|null
*/
public function getServer(): Server|WServer|HServer|null
{
return $this->server;
}
/**
* @param string $type
* @param string $host
* @param int $port
* @param int $mode
* @param array $settings
* @throws ReflectionException
* @throws ConfigException
* @throws Exception
*/
public function addListener(string $type, string $host, int $port, int $mode, array $settings = [])
{
if (!$this->server) {
$this->createBaseServer($type, $host, $port, $mode, $settings);
} else {
if (!isset($settings['settings'])) {
$settings['settings'] = [];
}
$this->addNewListener($type, $host, $port, $mode, $settings);
}
}
/**
* @throws ReflectionException
* @throws ConfigException
*/
public function initBaseServer($configs, int $daemon = 0): void
{
$context = di(ServerManager::class);
foreach ($this->sortService($configs['ports']) as $config) {
$this->startListenerHandler($context, $config, $daemon);
}
$this->bindCallback([Constant::PIPE_MESSAGE => [OnPipeMessage::class, 'onPipeMessage']]);
}
/**
* @return bool
* @throws ConfigException
* @throws Exception
*/
public function isRunner(): bool
{
$configs = Config::get('server', [], true);
foreach ($this->sortService($configs['ports']) as $config) {
if (checkPortIsAlready($config['port'])) {
return true;
}
}
return false;
}
/**
* @param string|OnProcessInterface|BaseProcess $customProcess
* @throws Exception
*/
public function addProcess(string|OnProcessInterface|BaseProcess $customProcess)
{
if (is_string($customProcess)) {
$customProcess = Kiri::getDi()->get($customProcess);
}
$system = sprintf('[%s].process', Config::get('id', 'system-service'));
$process = new Process(function (Process $process) use ($customProcess, $system) {
if (Kiri::getPlatform()->isLinux()) {
$process->name($system . '(' . $customProcess->getName() . ')');
}
$customProcess->process($process);
}, $customProcess->getRedirectStdinAndStdout(), $customProcess->getPipeType(), $customProcess->isEnableCoroutine());
$this->logger->debug($system . ' ' . $customProcess->getName() . ' start.');
$this->container->setBindings($customProcess->getName(), $process);
$this->server->addProcess($process);
}
/**
* @return array
*/
public function getSetting(): array
{
return $this->server->setting;
}
/**
* @param string $key
* @param string|int $value
*/
public static function setEnv(string $key, string|int $value): void
{
putenv(sprintf('%s=%s', $key, (string)$value));
}
/**
* @param ServerManager $context
* @param array $config
* @param int $daemon
* @throws ConfigException
* @throws ReflectionException
* @throws Exception
*/
private function startListenerHandler(ServerManager $context, array $config, int $daemon = 0)
{
if (!$this->server) {
$config = $this->mergeConfig($config, $daemon);
}
$context->addListener(
$config['type'], $config['host'], $config['port'], $config['mode'],
$config);
}
/**
* @param $config
* @param $daemon
* @return array
* @throws Exception
*/
private function mergeConfig($config, $daemon): array
{
$config['settings'] = $config['settings'] ?? [];
$config['settings']['daemonize'] = $daemon;
if (!isset($config['settings']['log_file'])) {
$config['settings']['log_file'] = storage('system.log');
}
$config['settings']['pid_file'] = storage('.swoole.pid');
$config['settings'][Constant::OPTION_ENABLE_REUSE_PORT] = true;
$config['events'] = $config['events'] ?? [];
return $config;
}
/**
* @param string $type
* @param string $host
* @param int $port
* @param int $mode
* @param array $settings
* @throws Exception
*/
private function addNewListener(string $type, string $host, int $port, int $mode, array $settings = [])
{
$id = Config::get('id', 'system-service');
$this->logger->debug(sprintf('[%s]' . $type . ' service %s::%d start', $id, $host, $port));
/** @var Server\Port $service */
$this->ports[$port] = $this->server->addlistener($host, $port, $mode);
if ($this->ports[$port] === false) {
throw new Exception("The port is already in use[$host::$port]");
}
if ($type == Constant::SERVER_TYPE_HTTP && !isset($settings['settings']['open_http_protocol'])) {
$settings['settings']['open_http_protocol'] = true;
if (in_array($this->server->setting['dispatch_mode'], [2, 4])) {
$settings['settings']['open_http2_protocol'] = true;
}
}
if ($type == Constant::SERVER_TYPE_WEBSOCKET && !isset($settings['settings']['open_websocket_protocol'])) {
$settings['settings']['open_websocket_protocol'] = true;
}
$this->ports[$port]->set($settings['settings'] ?? []);
$this->addServiceEvents($settings['events'] ?? [], $this->ports[$port]);
}
/**
* @param string $type
* @param string $host
* @param int $port
* @param int $mode
* @param array $settings
* @throws ReflectionException
* @throws ConfigException
* @throws Exception
*/
private function createBaseServer(string $type, string $host, int $port, int $mode, array $settings = [])
{
$match = match ($type) {
Constant::SERVER_TYPE_BASE, Constant::SERVER_TYPE_TCP,
Constant::SERVER_TYPE_UDP => Server::class,
Constant::SERVER_TYPE_HTTP => HServer::class,
Constant::SERVER_TYPE_WEBSOCKET => WServer::class
};
$this->server = new $match($host, $port, SWOOLE_PROCESS, $mode);
$this->server->set(array_merge(Config::get('server.settings', []), $settings['settings']));
$id = Config::get('id', 'system-service');
$this->logger->debug(sprintf('[%s]' . $type . ' service %s::%d start', $id, $host, $port));
$this->addDefaultListener($settings);
}
/**
* @param int $port
* @throws Exception
*/
public function stopServer(int $port)
{
if (!($pid = checkPortIsAlready($port))) {
return;
}
while (checkPortIsAlready($port)) {
Process::kill($pid, SIGTERM);
usleep(300);
}
}
/**
* @param array $settings
* @throws ReflectionException
* @throws Exception
*/
private function addDefaultListener(array $settings): void
{
if (($this->server->setting['task_worker_num'] ?? 0) > 0) {
$this->addTaskListener($settings['events']);
}
$this->container->setBindings(SwooleServerInterface::class, $this->server);
$this->addServiceEvents(ServerManager::DEFAULT_EVENT, $this->server);
if (!empty($settings['events']) && is_array($settings['events'])) {
$this->addServiceEvents($settings['events'], $this->server);
}
}
/**
* @param array $events
* @param Server|Port $server
*/
private function addServiceEvents(array $events, Server|Port $server)
{
foreach ($events as $name => $event) {
if (is_array($event) && is_string($event[0])) {
$event[0] = $this->container->get($event[0]);
}
$server->on($name, $event);
}
}
/**
*
*/
public function start()
{
$this->server->start();
}
/**
* @param string $class
* @return object
*/
private function getNewInstance(string $class): object
{
return $this->container->create($class);
}
/**
* @param OnTaskInterface|string $handler
* @param array $params
* @param int|null $workerId
* @throws ReflectionException
* @throws Exception
*/
public function task(OnTaskInterface|string $handler, array $params = [], int $workerId = null)
{
if ($workerId === null || $workerId <= $this->server->setting['worker_num']) {
$workerId = random_int($this->server->setting['worker_num'] + 1,
$this->server->setting['worker_num'] + 1 + $this->server->setting['task_worker_num']);
}
if (is_string($handler)) {
$implements = $this->container->getReflect($handler);
if (!in_array(OnTaskInterface::class, $implements->getInterfaceNames())) {
throw new Exception('Task must instance ' . OnTaskInterface::class);
}
$handler = $implements->newInstanceArgs($params);
}
$this->server->task(serialize($handler), $workerId);
}
/**
* @param mixed $message
* @param int $workerId
* @return mixed
*/
public function sendMessage(mixed $message, int $workerId): mixed
{
return $this->server?->sendMessage($message, $workerId);
}
/**
* @param array $events
* @throws ReflectionException
*/
private function addTaskListener(array $events = []): void
{
$task_use_object = $this->server->setting['task_object'] ?? $this->server->setting['task_use_object'] ?? false;
$reflect = $this->container->getReflect(OnServerTask::class)?->newInstance();
$this->server->on('finish', $events[Constant::FINISH] ?? [$reflect, 'onFinish']);
if ($task_use_object || $this->server->setting['task_enable_coroutine']) {
$this->server->on('task', $events[Constant::TASK] ?? [$reflect, 'onCoroutineTask']);
} else {
$this->server->on('task', $events[Constant::TASK] ?? [$reflect, 'onTask']);
}
}
/**
* @param array|null $settings
*/
public function bindCallback(?array $settings = [])
{
// TODO: Implement bindCallback() method.
if (count($settings) < 1) {
return;
}
foreach ($settings as $event_type => $callback) {
if ($this->server->getCallback($event_type) !== null) {
continue;
}
if (is_array($callback) && !is_object($callback[0])) {
$callback[0] = $this->container->get($callback[0]);
}
$this->server->on($event_type, $callback);
}
}
}
+28 -34
View File
@@ -1,34 +1,28 @@
<?php
declare(strict_types=1);
namespace Server;
use Exception;
use Kiri\Abstracts\Providers;
use Kiri\Application;
use Kiri\Kiri;
/**
* Class DatabasesProviders
* @package Database
*/
class ServerProviders extends Providers
{
/**
* @param Application $application
* @throws Exception
*/
public function onImport(Application $application)
{
$application->set('server', ['class' => Server::class]);
$container = Kiri::getDi();
$console = $container->get(\Symfony\Component\Console\Application::class);
$console->add($container->get(ServerCommand::class));
}
}
<?php
declare(strict_types=1);
namespace Kiri\Server;
use Kiri\Abstracts\Providers;
use Symfony\Component\Console\Application;
/**
* Class DatabasesProviders
* @package Database
*/
class ServerProviders extends Providers
{
/**
* @throws
*/
public function onImport(): void
{
$server = $this->container->get(ServerCommand::class);
$console = $this->container->get(Application::class);
$console->add($server);
}
}
+59
View File
@@ -0,0 +1,59 @@
<?php
namespace Kiri\Server;
use Kiri\Abstracts\Component;
use Kiri\Server\Abstracts\TraitServer;
use Swoole\Process;
use function config;
class State extends Component
{
use TraitServer;
public array $servers = [];
/**
* @return void
*/
public function init(): void
{
$this->servers = config('server.ports');
}
/**
* @return bool
* @throws
*/
public function isRunner(): bool
{
$ports = $this->sortService($this->servers);
foreach ($ports as $config) {
if (checkPortIsAlready($config['port'])) {
return true;
}
}
return false;
}
/**
* @param $port
* @throws
*/
public function exit($port): void
{
if (!($pid = checkPortIsAlready($port))) {
return;
}
while (checkPortIsAlready($port)) {
Process::kill($pid, 0) && Process::kill($pid, SIGTERM);
usleep(300);
}
}
}
-14
View File
@@ -1,14 +0,0 @@
<?php
namespace Server;
use Swoole\Server;
/**
* @mixin Server
*/
interface SwooleServerInterface
{
}
+17
View File
@@ -0,0 +1,17 @@
<?php
namespace Kiri\Server\Task;
class OnTaskFinish
{
/**
* @param int $task_id
* @param mixed $data
*/
public function __construct(int $task_id, mixed $data)
{
}
}
+95
View File
@@ -0,0 +1,95 @@
<?php
namespace Kiri\Server\Task;
use Kiri;
use Kiri\Router\Base\ExceptionHandlerDispatcher;
use Kiri\Router\DataGrip;
use Kiri\Router\Interface\ExceptionHandlerInterface;
use Kiri\Server\Constant;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\ContainerInterface;
use Psr\Container\NotFoundExceptionInterface;
use Psr\Http\Message\ResponseInterface;
use Swoole\Server;
/**
*
*/
class Task
{
public ExceptionHandlerInterface $exception;
/**
* @param ContainerInterface $container
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
public function __construct(public ContainerInterface $container)
{
$exception = \config('exception.task');
if (!in_array(ExceptionHandlerInterface::class, class_implements($exception))) {
$exception = ExceptionHandlerDispatcher::class;
}
$this->exception = $this->container->get($exception);
}
/**
* @param Server $server
* @return void
*/
public function initTaskWorker(Server $server): void
{
if (!isset($server->setting[Constant::OPTION_TASK_WORKER_NUM])) {
return;
}
if ($server->setting[Constant::OPTION_TASK_WORKER_NUM] < 1) {
return;
}
$server->on('finish', [$this, 'onFinish']);
$server->on('task', [$this, 'onTask']);
}
/**
* @param Server $server
* @param int $task_id
* @param mixed $data
* @return void
* @throws
*/
public function onFinish(Server $server, int $task_id, mixed $data): void
{
event(new OnTaskFinish($task_id, $data));
}
/**
* @param Server $server
* @param int $task_id
* @param int $src_worker_id
* @param mixed $data
* @return mixed
* @throws
*/
public function onTask(Server $server, int $task_id, int $src_worker_id, mixed $data): mixed
{
try {
$data = json_decode($data, true);
if (is_null($data)) {
return null;
}
$data[0] = Kiri::getDi()->get($data[0]);
return call_user_func($data, $task_id, $src_worker_id);
} catch (\Throwable $throwable) {
return $this->exception->emit($throwable, response());
}
}
}
+54
View File
@@ -0,0 +1,54 @@
<?php
namespace Kiri\Server\Task;
use Kiri\Di\Inject\Container;
use Kiri\Server\ServerInterface;
class TaskExecute implements TaskInterface
{
/**
* @var ServerInterface
*/
#[Container(ServerInterface::class)]
public ServerInterface $server;
/**
* @param mixed $data
* @param float $timeout
* @param int $dstWorkerId
* @return mixed
*/
public function taskWait(mixed $data, float $timeout = 0.5, int $dstWorkerId = -1): mixed
{
return $this->server->taskwait($data, $timeout, $dstWorkerId);
}
/**
* @param array $tasks
* @param float $timeout
* @return false|array
*/
public function taskCo(array $tasks, float $timeout = 0.5): false|array
{
return $this->server->taskCo($tasks, $timeout);
}
/**
* @param array $tasks
* @param float $timeout
* @return false|array
*/
public function taskWaitMulti(array $tasks, float $timeout = 0.5): false|array
{
return $this->server->taskWaitMulti($tasks, $timeout);
}
}
+31
View File
@@ -0,0 +1,31 @@
<?php
namespace Kiri\Server\Task;
interface TaskInterface
{
/**
* @param array $tasks
* @param float $timeout
* @return false|array
*/
public function taskWaitMulti(array $tasks, float $timeout = 0.5): false|array;
/**
* @param array $tasks
* @param float $timeout
* @return false|array
*/
public function taskCo(array $tasks, float $timeout = 0.5): false|array;
/**
* @param mixed $data
* @param float $timeout
* @param int $dstWorkerId
* @return mixed
*/
public function taskWait(mixed $data, float $timeout = 0.5, int $dstWorkerId = -1): mixed;
}
-32
View File
@@ -1,32 +0,0 @@
<?php
namespace Server;
trait TraitServer
{
/**
* @param array $ports
* @return array
*/
public function sortService(array $ports): array
{
$array = [];
foreach ($ports as $port) {
if ($port['type'] == Constant::SERVER_TYPE_WEBSOCKET) {
array_unshift($array, $port);
} else if ($port['type'] == Constant::SERVER_TYPE_HTTP) {
if (!empty($array) && $array[0]['type'] == Constant::SERVER_TYPE_WEBSOCKET) {
$array[] = $port;
} else {
array_unshift($array, $port);
}
} else {
$array[] = $port;
}
}
return $array;
}
}
+24 -22
View File
@@ -1,25 +1,27 @@
{
"name": "game-worker/kiri-http-server",
"description": "http-server",
"license": "MIT",
"authors": [
{
"name": "向林",
"email": "as2252258@163.com"
}
],
"require": {
"php": ">=8.0",
"ext-json": "*",
"composer-runtime-api": "^2.0",
"psr/http-server-middleware": "^1.0",
"psr/http-message": "^1.0",
"psr/event-dispatcher": "^1.0",
"game-worker/kiri-http-message": "^v1.2"
},
"autoload": {
"psr-4": {
"Server\\": "./"
}
"name": "game-worker/kiri-http-server",
"description": "http-server",
"license": "MIT",
"authors": [
{
"name": "向林",
"email": "as2252258@163.com"
}
],
"require": {
"php": ">=8.0",
"ext-json": "*",
"composer-runtime-api": "^2.0",
"psr/http-server-middleware": "^1.0",
"psr/http-message": "^1.0",
"psr/event-dispatcher": "^1.0",
"ext-inotify": "*",
"ext-posix": "*",
"ext-pcntl": "*"
},
"autoload": {
"psr-4": {
"Kiri\\Server\\": "./"
}
}
}