Compare commits

..

489 Commits

Author SHA1 Message Date
as2252258 908c8719d3 eee 2026-06-24 20:17:54 +08:00
as2252258 69b845d924 eee 2026-06-12 23:57:18 +08:00
as2252258 d9c8344a29 eee 2026-04-17 16:31:59 +08:00
as2252258 ced9f89ebe eee 2026-04-17 14:10:47 +08:00
as2252258 ee487649fa eee 2026-04-17 13:56:30 +08:00
as2252258 0cdb100180 eee 2026-04-17 11:57:08 +08:00
as2252258 a9f1afb866 eee 2026-04-17 11:56:44 +08:00
as2252258 853a9661bc eee 2026-04-17 11:47:59 +08:00
as2252258 2d4951b0dd eee 2026-04-04 10:53:46 +08:00
as2252258 e819fe9ba4 eee 2026-04-04 10:41:53 +08:00
as2252258 9e1e16dca9 eee 2026-04-04 10:34:16 +08:00
as2252258 77cf882f2e eee 2026-02-26 14:39:04 +08:00
as2252258 eeb925b5b9 eee 2025-12-31 00:19:28 +08:00
as2252258 59042c0110 eee 2025-12-23 19:04:18 +08:00
as2252258 7c459cc9bc eee 2025-12-23 19:03:21 +08:00
as2252258 968cdbd11a eee 2025-12-18 15:39:40 +08:00
as2252258 37b59c8536 eee 2025-07-16 14:54:35 +08:00
as2252258 9376a73628 eee 2025-07-14 17:55:15 +08:00
as2252258 eebaf63999 eee 2025-07-14 15:34:37 +08:00
as2252258 e399837dad eee 2025-07-11 15:04:25 +08:00
as2252258 7b8a43f94f eee 2025-07-11 14:56:17 +08:00
as2252258 c0f8133926 eee 2025-07-11 14:51:53 +08:00
as2252258 f533d3262b eee 2025-07-11 14:44:06 +08:00
as2252258 c767a1745a eee 2025-07-11 11:50:54 +08:00
as2252258 251e05dbf7 eee 2024-12-27 22:34:17 +08:00
as2252258 5defd5fb30 eee 2024-12-27 21:53:16 +08:00
as2252258 95f52b58fd eee 2024-12-27 18:32:55 +08:00
as2252258 0a97c9b29a eee 2024-12-27 18:30:35 +08:00
as2252258 3336573b4d eee 2024-12-27 18:29:06 +08:00
as2252258 44187c2354 eee 2024-12-27 18:26:49 +08:00
as2252258 a058f3ed99 eee 2024-12-27 18:21:30 +08:00
as2252258 faff485006 eee 2024-12-27 17:59:52 +08:00
as2252258 e2b012126a eee 2024-12-27 17:58:52 +08:00
as2252258 b36e21d8ee eee 2024-11-19 09:12:45 +08:00
as2252258 d191296ecc eee 2024-11-19 09:12:29 +08:00
as2252258 03825dcb09 eee 2024-11-19 09:10:12 +08:00
as2252258 d500bb78c8 eee 2024-11-18 17:32:49 +08:00
as2252258 e2cac499b6 eee 2024-11-18 17:30:21 +08:00
as2252258 bb557a9aa9 eee 2024-11-18 17:15:22 +08:00
as2252258 89cc6a04eb eee 2024-11-18 17:09:12 +08:00
as2252258 91e5c4d66a eee 2024-11-18 17:07:57 +08:00
as2252258 7e5301f02a eee 2024-11-18 17:05:21 +08:00
as2252258 f76f4c02b6 eee 2024-11-18 16:59:10 +08:00
as2252258 2afe84694d eee 2024-11-18 16:55:55 +08:00
as2252258 8f80811da6 eee 2024-11-18 16:51:24 +08:00
as2252258 e46ad90c46 eee 2024-11-18 16:50:22 +08:00
as2252258 4172349872 eee 2024-11-18 16:46:31 +08:00
as2252258 0849870234 eee 2024-11-18 16:42:28 +08:00
as2252258 9fcd47ecd1 eee 2024-11-18 16:40:55 +08:00
as2252258 3bfcaf8236 eee 2024-11-18 16:39:59 +08:00
as2252258 23110643e0 eee 2024-11-18 16:35:37 +08:00
as2252258 d9980e9e1b eee 2024-11-18 16:05:10 +08:00
as2252258 a580607ec8 eee 2024-11-18 16:03:49 +08:00
as2252258 1e5fe2412a eee 2024-11-18 14:39:26 +08:00
as2252258 2be7134ea6 eee 2024-11-18 14:34:42 +08:00
as2252258 125dd9e5b2 eee 2024-11-18 14:31:26 +08:00
as2252258 7fa1110a50 eee 2024-11-18 14:29:47 +08:00
as2252258 b23c7eff83 eee 2024-11-18 14:29:19 +08:00
as2252258 2e86a73187 eee 2024-11-18 14:28:41 +08:00
as2252258 263b4d53eb eee 2024-11-18 14:19:31 +08:00
as2252258 ddf25de2c5 eee 2024-11-18 14:17:02 +08:00
as2252258 54c34ccf8f eee 2024-11-18 14:16:03 +08:00
as2252258 ad9f536c90 eee 2024-11-18 14:15:17 +08:00
as2252258 c068c2db4f eee 2024-11-18 14:14:15 +08:00
as2252258 c4bca4f88e eee 2024-11-18 14:13:47 +08:00
as2252258 f6a00d89c1 eee 2024-11-18 14:00:51 +08:00
as2252258 7f93ab046b eee 2024-11-18 13:53:50 +08:00
as2252258 64514b069e eee 2024-11-18 13:45:19 +08:00
as2252258 e669d80bb9 eee 2024-11-18 13:43:38 +08:00
as2252258 b3777a64e3 eee 2024-11-18 13:42:12 +08:00
as2252258 f858cecf1a eee 2024-11-18 13:40:23 +08:00
as2252258 e0eda4b42c eee 2024-11-18 12:23:58 +08:00
as2252258 b54d956a2d eee 2024-11-18 12:22:49 +08:00
as2252258 bce6870538 eee 2024-11-18 12:21:33 +08:00
as2252258 ab1a18a886 eee 2024-11-18 12:19:26 +08:00
as2252258 26bd43510e eee 2024-11-18 12:13:09 +08:00
as2252258 cb73717c7f eee 2024-11-18 12:11:11 +08:00
as2252258 7360a8107b eee 2024-11-18 12:07:24 +08:00
as2252258 cc98a527cd eee 2024-11-18 12:06:53 +08:00
as2252258 694bde36d9 eee 2024-11-18 11:50:13 +08:00
as2252258 bbae787d4b eee 2024-11-18 11:49:49 +08:00
as2252258 dc15054c86 eee 2024-11-18 11:46:39 +08:00
as2252258 787f035d62 eee 2024-11-18 11:45:31 +08:00
as2252258 09ec4eb9c3 eee 2024-11-18 11:43:08 +08:00
as2252258 60d08ccaca eee 2024-11-18 11:42:02 +08:00
as2252258 8c5eefb9a7 eee 2024-11-18 11:38:27 +08:00
as2252258 9cb6db5e26 eee 2024-11-18 11:32:40 +08:00
as2252258 82ff5c5d42 eee 2024-11-18 11:27:05 +08:00
as2252258 a5378658ee eee 2024-11-18 11:24:08 +08:00
as2252258 ac629ece15 eee 2024-09-29 09:01:39 +08:00
as2252258 42794168b5 eee 2024-09-29 09:01:03 +08:00
as2252258 e626b38d67 eee 2024-09-04 12:28:24 +08:00
as2252258 0e34ee9542 eee 2024-09-04 12:27:52 +08:00
as2252258 eaed9020db eee 2024-09-04 11:52:08 +08:00
as2252258 e1841d908b eee 2024-09-04 11:43:24 +08:00
as2252258 53bcb44b0e eee 2024-09-04 11:24:38 +08:00
as2252258 f7e5faa668 eee 2024-09-04 11:23:48 +08:00
as2252258 9b22f66571 eee 2024-09-04 11:21:12 +08:00
as2252258 90154ece54 eee 2024-09-04 10:51:16 +08:00
as2252258 e6c051ad24 eee 2024-09-04 10:28:59 +08:00
as2252258 b5937acf3c eee 2024-09-04 10:28:02 +08:00
as2252258 97d11abd4e eee 2024-09-04 10:25:34 +08:00
as2252258 133c05be44 eee 2024-09-04 10:22:53 +08:00
as2252258 f3244cce0f eee 2024-09-04 10:20:02 +08:00
as2252258 e4321de504 eee 2024-09-04 10:19:03 +08:00
as2252258 e1e44da5b2 eee 2024-09-04 10:14:29 +08:00
as2252258 a756532738 eee 2024-09-04 10:07:59 +08:00
as2252258 80b9ae8f1e eee 2024-09-04 09:40:31 +08:00
as2252258 0e1fcef911 eee 2024-09-03 15:05:18 +08:00
as2252258 92dd75b82c eee 2024-09-03 14:47:27 +08:00
as2252258 12a1aac9e5 eee 2024-08-29 18:06:58 +08:00
as2252258 917e82064a eee 2024-08-29 17:01:07 +08:00
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
54 changed files with 2299 additions and 1410 deletions
+188
View File
@@ -0,0 +1,188 @@
<?php
namespace Kiri\Server\Abstracts;
use Exception;
use Kiri\Abstracts\Component;
use Kiri\Error\StdoutLogger;
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\Processes\TraitProcess;
use Kiri\Server\ServerInterface;
use Kiri\Server\Task\Task;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\NotFoundExceptionInterface;
use Swoole\Server;
/**
*/
class AsyncServer extends Component implements ServerInterface
{
use TraitServer;
use TraitProcess;
/**
* @var Server|null
*/
private ?Server $server = null;
/**
* @param array $service
* @param int $daemon
* @return void
* @throws
*/
public function initCoreServers(array $service, int $daemon = 0): void
{
$service = $this->createBaseServer($this->genConfigService($service), $daemon);
if (isset($this->server->setting[Constant::OPTION_TASK_WORKER_NUM])) {
$this->container->get(Task::class)->initTaskWorker($this->server);
}
foreach ($this->_process as $process)
$this->server->addProcess($process);
foreach ($service as $value)
$this->addListener($value);
$this->provider->on(OnServerBeforeStart::class, [$this, 'onSignal']);
}
/**
* @return bool
* @throws
*/
public function shutdown(): bool
{
$this->server->shutdown();
$this->dispatch->dispatch(new OnShutdown);
return true;
}
/**
* @param array $service
* @param int $daemon
* @return array
* @throws ContainerExceptionInterface
* @throws NotFindClassException
* @throws NotFoundExceptionInterface
*/
private function createBaseServer(array $service, int $daemon = 0): array
{
$config = array_pop($service);
$match = $this->getServerClass($config->type);
if (is_null($match)) {
throw new NotFindClassException('Unknown server type ' . $config->type);
}
$this->container->get(StdoutLogger::class)->println('Listen address ' . $config->type . ' ' . $config->host . '::' . $config->port);
$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->event($this->server, array_merge(\config('servers.server.events', []), $config->events));
$this->container->bind(ServerInterface::class, $this->server);
return $service;
}
/**
* @param SConfig $config
* @param int $daemon
* @return array
* @throws
*/
protected function systemConfig(SConfig $config, int $daemon): array
{
$settings = array_merge(\config('servers.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());
}
$port->set($this->resetSettings($config->type, $config->settings));
$this->event($port, $config->getEvents());
$this->container->get(StdoutLogger::class)->println('Listen address ' . $config->type . ' ' . $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 event(Server\Port|Server $base, array $events): void
{
$container = $this->container;
foreach ($events as $name => $event) {
if (is_array($event) && is_string($event[0])) {
$event[0] = $container->get($event[0]);
}
$base->on($name, $event);
}
}
/**
* @return void
* @throws
*/
public function start(): void
{
$this->dispatch->dispatch(new OnServerBeforeStart);
$this->server->start();
}
}
-116
View File
@@ -1,116 +0,0 @@
<?php
namespace Kiri\Server\Abstracts;
use Kiri\Context;
use Kiri\Server\Broadcast\OnBroadcastInterface;
use Kiri\Server\Contract\OnProcessInterface;
use Swoole\Coroutine;
use Swoole\Process;
/**
*
*/
abstract class BaseProcess implements OnProcessInterface
{
protected bool $isStop = false;
protected bool $redirect_stdin_and_stdout = FALSE;
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 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 onProcessStop(): void
{
$this->isStop = true;
}
/**
*
*/
public function onSigterm(): static
{
if (!Context::inCoroutine()) {
Process::signal(SIGTERM, fn($data) => $this->onShutdown($data));
} else {
Coroutine::create(function () {
$data = Coroutine::waitSignal(SIGTERM, -1);
if ($data) {
$this->onShutdown($data);
}
});
}
return $this;
}
/**
* @param $data
*/
protected function onShutdown($data): void
{
$this->isStop = true;
$value = Context::getContext('waite:process:message');
if (Coroutine::exists($value)) {
Coroutine::cancel($value);
}
}
}
+446
View File
@@ -0,0 +1,446 @@
<?php
namespace Kiri\Server\Abstracts;
use Kiri\Di\HotReloadState;
use Kiri\Di\Inject\Container;
use Kiri\Error\StdoutLogger;
use Kiri\Events\EventProvider;
use Kiri\Router\Router;
use Kiri\Server\Events\OnWorkerStart;
use Kiri\Server\Processes\AbstractProcess;
use Kiri\Server\Processes\OnProcessInterface;
use Kiri\Server\ServerInterface;
use Psr\Log\LoggerInterface;
use Swoole\Event;
use Swoole\Process;
class FileWatcher extends AbstractProcess implements OnProcessInterface
{
protected mixed $pipe;
#[Container(LoggerInterface::class)]
public StdoutLogger|LoggerInterface $logger;
protected bool $enable_coroutine = false;
protected array $watches = [];
protected bool $enable_queue = false;
protected bool $reloading = false;
private array $watchPaths = [];
private array $excludePatterns = [];
private array $extensions = ['php'];
private string $strategy;
private bool $running = false;
private $inotifyFd = null;
private array $inotifyWatchMap = [];
private $fswatchProcess = null;
private array $fswatchPipes = [];
private int $pollInterval = 2;
private array $fileSnapshot = [];
private $pollTimer = null;
private $debounceTimer = null;
private int $debounceMs = 200;
private const STRATEGY_INOTIFY = 'inotify';
private const STRATEGY_FSWATCH = 'fswatch';
private const STRATEGY_POLL = 'poll';
public function __construct()
{
$this->watchPaths = config('servers.reload.listen', []);
$this->excludePatterns = config('servers.reload.scan.skip_patterns', []);
$this->extensions = config('servers.reload.scan.extensions', ['php']);
di(EventProvider::class)->on(OnWorkerStart::class, [di(Router::class), 'scan_build_route']);
$this->strategy = $this->detectBestStrategy();
}
public function getName(): string
{
return 'hotReload';
}
public function onSigterm(): void
{
$this->stop();
}
private function detectBestStrategy(): string
{
$forced = config('servers.reload.scan.strategy', 'auto');
if (in_array($forced, [self::STRATEGY_INOTIFY, self::STRATEGY_FSWATCH, self::STRATEGY_POLL], true)) {
return $forced;
}
if ($this->isMountedWindowsPath()) {
return $this->isFswatchAvailable() ? self::STRATEGY_FSWATCH : self::STRATEGY_POLL;
}
if (extension_loaded('inotify')) {
return self::STRATEGY_INOTIFY;
}
if ($this->isFswatchAvailable()) {
return self::STRATEGY_FSWATCH;
}
return self::STRATEGY_POLL;
}
private function isMountedWindowsPath(): bool
{
foreach ($this->watchPaths as $path) {
$real = realpath($path) ?: (string)$path;
$real = str_replace('\\', '/', $real);
if (str_starts_with($real, '/mnt/')) {
return true;
}
}
return false;
}
private function isFswatchAvailable(): bool
{
$output = [];
$returnVar = 0;
exec('which fswatch 2>/dev/null', $output, $returnVar);
return $returnVar === 0 && !empty($output[0]);
}
public function setDebounce(int $milliseconds): self
{
$this->debounceMs = $milliseconds;
return $this;
}
public function setPollInterval(int $seconds): self
{
$this->pollInterval = $seconds;
return $this;
}
public function process(Process|null $process): void
{
if ($this->running) {
return;
}
$this->running = true;
switch ($this->strategy) {
case self::STRATEGY_INOTIFY:
$this->startInotify();
break;
case self::STRATEGY_FSWATCH:
$this->startFswatch();
break;
case self::STRATEGY_POLL:
$this->startPolling();
break;
}
Event::wait();
}
public function stop(): void
{
$this->running = false;
if ($this->inotifyFd && is_resource($this->inotifyFd)) {
@Event::del($this->inotifyFd);
@fclose($this->inotifyFd);
$this->inotifyFd = null;
}
if (isset($this->fswatchPipes[1]) && is_resource($this->fswatchPipes[1])) {
@Event::del($this->fswatchPipes[1]);
@fclose($this->fswatchPipes[1]);
}
if (isset($this->fswatchPipes[2]) && is_resource($this->fswatchPipes[2])) {
@fclose($this->fswatchPipes[2]);
}
if ($this->fswatchProcess && is_resource($this->fswatchProcess)) {
@proc_terminate($this->fswatchProcess);
@proc_close($this->fswatchProcess);
$this->fswatchProcess = null;
}
if ($this->pollTimer) {
\Swoole\Timer::clear($this->pollTimer);
$this->pollTimer = null;
}
if ($this->debounceTimer) {
\Swoole\Timer::clear($this->debounceTimer);
$this->debounceTimer = null;
}
@Event::exit();
}
private function isExcluded(string $path): bool
{
foreach ($this->excludePatterns as $pattern) {
if (strpos($path, $pattern) !== false) {
return true;
}
}
return false;
}
private function hasValidExtension(string $path): bool
{
if (empty($this->extensions)) {
return true;
}
$ext = pathinfo($path, PATHINFO_EXTENSION);
return in_array($ext, $this->extensions, true);
}
private function triggerCallback(array $changedFiles): void
{
$changedFiles = array_values(array_unique(array_filter($changedFiles, function ($file) {
return !$this->isExcluded($file) && $this->hasValidExtension($file);
})));
if (empty($changedFiles)) {
return;
}
if ($this->debounceTimer) {
\Swoole\Timer::clear($this->debounceTimer);
}
$this->debounceTimer = \Swoole\Timer::after($this->debounceMs, function () use ($changedFiles) {
$this->debounceTimer = null;
$this->reload($changedFiles);
});
}
private function reload(array $changedFiles): void
{
if ($this->reloading) {
return;
}
$this->reloading = true;
try {
$preview = implode(', ', array_slice($changedFiles, 0, 3));
if (count($changedFiles) > 3) {
$preview .= ' ...';
}
di(HotReloadState::class)->store($changedFiles);
di(StdoutLogger::class)->println('detected file changes, reloading server: ' . $preview);
$server = di(ServerInterface::class);
if (method_exists($server, 'reload')) {
$server->reload();
}
} finally {
$this->reloading = false;
}
}
private function startInotify(): void
{
$this->inotifyFd = inotify_init();
stream_set_blocking($this->inotifyFd, false);
foreach ($this->watchPaths as $path) {
$this->addInotifyWatchRecursive($path);
}
Event::add($this->inotifyFd, function ($fd) {
$events = inotify_read($fd);
if ($events === false) {
return;
}
$changedFiles = [];
foreach ($events as $event) {
$wd = $event['wd'];
$dir = $this->inotifyWatchMap[$wd] ?? '';
$name = $event['name'] ?? '';
$filePath = $name === '' ? $dir : $dir . '/' . $name;
if (($event['mask'] & IN_CREATE) && $filePath !== '' && is_dir($filePath) && !$this->isExcluded($filePath)) {
$this->addInotifyWatchRecursive($filePath);
}
if ($filePath !== '') {
$changedFiles[] = $filePath;
}
}
$this->triggerCallback($changedFiles);
});
}
private function addInotifyWatchRecursive(string $path): void
{
if ($this->isExcluded($path)) {
return;
}
if (is_file($path)) {
if ($this->hasValidExtension($path)) {
$wd = inotify_add_watch($this->inotifyFd, $path, IN_MODIFY | IN_CREATE | IN_DELETE | IN_MOVE | IN_CLOSE_WRITE);
$this->inotifyWatchMap[$wd] = dirname($path);
}
return;
}
if (!is_dir($path)) {
return;
}
$wd = inotify_add_watch($this->inotifyFd, $path, IN_MODIFY | IN_CREATE | IN_DELETE | IN_MOVE | IN_CLOSE_WRITE);
$this->inotifyWatchMap[$wd] = $path;
$items = scandir($path);
foreach ($items as $item) {
if ($item === '.' || $item === '..') {
continue;
}
$full = $path . '/' . $item;
if (is_dir($full)) {
$this->addInotifyWatchRecursive($full);
}
}
}
private function startFswatch(): void
{
$descriptorspec = [
0 => ['pipe', 'r'],
1 => ['pipe', 'w'],
2 => ['pipe', 'w'],
];
$excludeArgs = [];
foreach ($this->excludePatterns as $pattern) {
$excludeArgs[] = '--exclude';
$excludeArgs[] = '.*' . preg_quote($pattern, '/') . '.*';
}
$filterArgs = [];
foreach ($this->extensions as $ext) {
$filterArgs[] = '--include';
$filterArgs[] = '\.' . $ext . '$';
}
$cmd = 'fswatch -0 -r ' . implode(' ', array_merge($excludeArgs, $filterArgs)) . ' ' . implode(' ', array_map('escapeshellarg', $this->watchPaths));
$this->fswatchProcess = proc_open($cmd, $descriptorspec, $this->fswatchPipes);
if (!is_resource($this->fswatchProcess)) {
throw new \RuntimeException('Failed to start fswatch process');
}
stream_set_blocking($this->fswatchPipes[1], false);
Event::add($this->fswatchPipes[1], function ($pipe) {
$data = fread($pipe, 8192);
if ($data === false || $data === '') {
return;
}
$files = explode("\0", trim($data, "\0"));
$changedFiles = array_values(array_filter($files, function ($file) {
return $file !== '';
}));
$this->triggerCallback($changedFiles);
});
}
private function startPolling(): void
{
$this->buildFileSnapshot();
$intervalMs = $this->pollInterval * 1000;
$this->pollTimer = \Swoole\Timer::tick($intervalMs, function () {
$newSnapshot = [];
$changedFiles = $this->compareSnapshots($newSnapshot);
if (!empty($changedFiles)) {
$this->triggerCallback($changedFiles);
}
$this->fileSnapshot = $newSnapshot;
});
}
private function buildFileSnapshot(): void
{
$this->fileSnapshot = [];
foreach ($this->watchPaths as $path) {
$this->scanDirectory($path, $this->fileSnapshot);
}
}
private function scanDirectory(string $path, array &$snapshot): void
{
if ($this->isExcluded($path)) {
return;
}
if (is_file($path)) {
if ($this->hasValidExtension($path)) {
$snapshot[$path] = filemtime($path);
}
return;
}
if (!is_dir($path)) {
return;
}
$items = scandir($path);
foreach ($items as $item) {
if ($item === '.' || $item === '..') {
continue;
}
$full = $path . '/' . $item;
$this->scanDirectory($full, $snapshot);
}
}
private function compareSnapshots(array &$newSnapshot): array
{
$changed = [];
foreach ($this->watchPaths as $path) {
$this->scanDirectory($path, $newSnapshot);
}
foreach ($newSnapshot as $file => $mtime) {
if (!isset($this->fileSnapshot[$file])) {
$changed[] = $file;
} elseif ($this->fileSnapshot[$file] !== $mtime) {
$changed[] = $file;
}
}
foreach ($this->fileSnapshot as $file => $mtime) {
if (!isset($newSnapshot[$file])) {
$changed[] = $file;
}
}
return $changed;
}
}
+171
View File
@@ -0,0 +1,171 @@
<?php
namespace Kiri\Server\Abstracts;
use Kiri\Di\Inject\Container;
use Kiri\Error\StdoutLogger;
use Kiri\Events\EventProvider;
use Kiri\Router\Router;
use Kiri\Server\Events\OnWorkerStart;
use Kiri\Server\ServerInterface;
use Psr\Log\LoggerInterface;
use Swoole\Event;
use Swoole\Process;
use Kiri\Server\Processes\AbstractProcess;
class HotReload extends AbstractProcess
{
/**
* @var mixed
*/
protected mixed $pipe;
/**
* @var LoggerInterface|StdoutLogger
*/
#[Container(LoggerInterface::class)]
public StdoutLogger|LoggerInterface $logger;
/**
* @var bool
*/
protected bool $enable_coroutine = false;
/**
* @var array
*/
protected array $watches = [];
/**
* @var bool
*/
protected bool $enable_queue = false;
/**
* @var bool
*/
protected bool $reloading = false;
/**
* @throws \Exception
*/
public function __construct()
{
di(EventProvider::class)->on(OnWorkerStart::class, [di(Router::class), 'scan_build_route']);
}
/**
* @return string
*/
public function getName(): string
{
return 'hotReload';
}
/**
* @return void
*/
public function onSigterm(): void
{
// TODO: Implement onSigterm() method.
$this->stop();
}
/**
* @param ?Process $process
*/
public function process(Process|null $process): void
{
$this->pipe = inotify_init();
$this->addListen();
Event::add($this->pipe, function () use ($process) {
$read = inotify_read($this->pipe);
if (count($read) > 0) {
$this->reload();
}
});
Event::cycle(function (): void {
if ($this->isStop()) {
Event::exit();
}
});
Event::wait();
}
/**
* @return void
*/
public function reload(): void
{
if ($this->reloading) {
return;
}
$this->reloading = true;
di(StdoutLogger::class)->println('reloading server[' . \config('site.id', 'system-service') . '], please waite.');
$this->clear();
di(ServerInterface::class)->reload();
$this->addListen();
$this->reloading = false;
}
/**
* @return void
*/
protected function addListen(): void
{
foreach (config('servers.reload.listen') as $value) {
$this->readDirectory($value);
}
}
/**
* @return void
*/
protected function clear(): void
{
$this->watches = [];
}
/**
* @param string $directory
* @return void
*/
public function readFile(string $directory): void
{
if (str_ends_with($directory, '.php') === true) {
inotify_add_watch($this->pipe, $directory, IN_MODIFY | IN_MOVE | IN_CREATE | IN_DELETE);
}
}
/**
* @param string $directory
* @return void
*/
public function readDirectory(string $directory): void
{
foreach (glob($directory . '/*') as $data) {
if (is_dir($data)) {
$this->readDirectory($data);
} else {
$this->readFile($data);
}
}
}
}
+12 -33
View File
@@ -4,13 +4,7 @@
namespace Kiri\Server\Abstracts;
use Kiri\Annotation\Inject;
use Exception;
use Kiri\Abstracts\Config;
use Kiri\Exception\ConfigException;
use Kiri;
use Psr\Log\LoggerInterface;
use Kiri\Error\StdoutLogger;
/**
* Class Server
@@ -19,37 +13,22 @@ use Psr\Log\LoggerInterface;
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
* @throws
*/
public function __construct()
{
}
/**
* @return StdoutLogger
*/
protected function getLogger(): StdoutLogger
{
return \Kiri::getLogger();
}
}
+132
View File
@@ -0,0 +1,132 @@
<?php
namespace Kiri\Server\Abstracts;
use Exception;
use Kiri;
use Swoole\Http\Server as HServer;
use Swoole\Server;
use Kiri\Server\Processes\AbstractProcess;
use Kiri\Server\Constant;
use Kiri\Server\Config;
use Swoole\WebSocket\Server as WServer;
use Swoole\Process;
trait TraitServer
{
/**
* @return void
* @throws
*/
public function onSignal(): void
{
$signal = \config('servers.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 {
Kiri::getLogger()->alert('Pid ' . getmypid() . ' get signo ' . $no);
$this->shutdown();
} catch (\Throwable $exception) {
\Kiri::getLogger()->json_log($exception);
}
}
/**
* @param $signal
* @param $callback
* @return void
*/
private function onPcntlSignal($signal, $callback): void
{
\pcntl_signal($signal, $callback);
}
/**
* @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
};
}
}
-34
View File
@@ -1,34 +0,0 @@
<?php
namespace Kiri\Server\Broadcast;
use Kiri;
use Kiri\Server\ProcessManager;
use Kiri\Server\SwooleServerInterface;
class Broadcast
{
/**
* @param $message
* @return void
*/
public function broadcast($message)
{
$di = Kiri::getDi();
$di->get(ProcessManager::class)->push($message);
$server = $di->get(SwooleServerInterface::class);
$total = $server->setting['worker_num'] + $server->setting['task_worker_num'];
for ($i = 0; $i < $total; $i++) {
if ($i == env('environmental_workerId')) {
continue;
}
$server->sendMessage(serialize(new Message($message)), $i);
}
}
}
-34
View File
@@ -1,34 +0,0 @@
<?php
namespace Kiri\Server\Broadcast;
use Kiri;
use Kiri\Server\Contract\OnPipeMessageInterface;
use Psr\Log\LoggerInterface;
/**
*
*/
class Message implements OnPipeMessageInterface, OnBroadcastInterface
{
/**
* @param mixed $data
*/
public function __construct(public mixed $data)
{
}
/**
* @return void
*/
public function process(): void
{
$logger = Kiri::getDi()->get(LoggerInterface::class);
$logger->debug(env('environmental') . '::' . env('environmental_workerId', 0) . '::' . $this->data);
}
}
-14
View File
@@ -1,14 +0,0 @@
<?php
namespace Kiri\Server\Broadcast;
interface OnBroadcastInterface
{
/**
* @return void
*/
public function process(): void;
}
+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;
}
}
+111 -111
View File
@@ -11,123 +11,123 @@ namespace Kiri\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 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 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 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 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 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 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';
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';
}
+5 -3
View File
@@ -4,7 +4,6 @@ namespace Kiri\Server\Contract;
use Swoole\Server;
/**
*
*/
@@ -13,9 +12,12 @@ interface OnCloseInterface
/**
* @param int $fd
* @param Server $server
* @param int $fd
* @param int $reactorId
* @return void
*/
public function onClose(int $fd): void;
public function onClose(Server $server, int $fd, int $reactorId): void;
}
+3 -2
View File
@@ -10,9 +10,10 @@ interface OnConnectInterface
/**
* @param Server $server
* @param int $fd
* @param int $fd
* @param int $reactorId
* @return void
*/
public function onConnect(Server $server, int $fd): void;
public function onConnect(Server $server, int $fd, int $reactorId): void;
}
+18 -18
View File
@@ -1,18 +1,18 @@
<?php
namespace Kiri\Server\Contract;
use Swoole\Server;
interface OnDisconnectInterface
{
/**
* @param int $fd
*/
public function onDisconnect(int $fd): void;
}
<?php
namespace Kiri\Server\Contract;
use Swoole\{WebSocket\Server};
interface OnDisconnectInterface
{
/**
* @param Server $server
* @param int $fd
* @return void
*/
public function onDisconnect(Server $server, int $fd): void;
}
+19 -22
View File
@@ -1,22 +1,19 @@
<?php
namespace Kiri\Server\Contract;
use Swoole\Http\Request;
use Swoole\Http\Response;
/**
*
*/
interface OnHandshakeInterface
{
/**
* @param Request $request
* @param Response $response
*/
public function onHandshake(Request $request, Response $response): void;
}
<?php
namespace Kiri\Server\Contract;
use Swoole\Http\Request;
use Swoole\Http\Response;
interface OnHandshakeInterface
{
/**
* @param Request $request
* @param Response $response
* @return void
*/
public function onHandshake(Request $request, Response $response): void;
}
+19 -18
View File
@@ -1,18 +1,19 @@
<?php
namespace Kiri\Server\Contract;
use Kiri\Websocket\WebSocketInterface;
use Swoole\WebSocket\Frame;
interface OnMessageInterface
{
/**
* @param Frame $frame
* @return void
*/
public function onMessage(Frame $frame): void;
}
<?php
namespace Kiri\Server\Contract;
use Swoole\WebSocket\Server;
use Swoole\WebSocket\Frame;
interface OnMessageInterface
{
/**
* @param Server $server
* @param Frame $frame
* @return void
*/
public function onMessage(Server $server, Frame $frame): void;
}
+19 -16
View File
@@ -1,16 +1,19 @@
<?php
namespace Kiri\Server\Contract;
use Swoole\Http\Request;
interface OnOpenInterface
{
/**
* @param Request $request
*/
public function onOpen(Request $request): void;
}
<?php
namespace Kiri\Server\Contract;
use Swoole\Http\Request;
use Swoole\WebSocket\Server;
interface OnOpenInterface
{
/**
* @param Server $server
* @param Request $request
* @return void
*/
public function onOpen(Server $server, Request $request): void;
}
+2 -2
View File
@@ -2,7 +2,7 @@
namespace Kiri\Server\Contract;
use Kiri\Server\Abstracts\Server;
use Swoole\Server;
interface OnPacketInterface
{
@@ -12,7 +12,7 @@ interface OnPacketInterface
* @param Server $server
* @param string $data
* @param array $clientInfo
* @return mixed
* @return void
*/
public function onPacket(Server $server, string $data, array $clientInfo): void;
+7 -2
View File
@@ -4,6 +4,8 @@
namespace Kiri\Server\Contract;
use Swoole\Server;
/**
* Interface OnPipeMessageInterface
* @package Server\Contract
@@ -12,9 +14,12 @@ interface OnPipeMessageInterface
{
/**
*
* @param Server $server
* @param int $src_worker_id
* @param mixed $message
* @return void
*/
public function process(): void;
public function onPipeMessage(Server $server, int $src_worker_id, mixed $message): void;
-25
View File
@@ -1,25 +0,0 @@
<?php
namespace Kiri\Server\Contract;
use Swoole\Process;
/**
* Interface BaseProcess
* @package Contract
*/
interface OnProcessInterface
{
/**
* @param Process $process
*/
public function process(Process $process): void;
}
+8 -8
View File
@@ -12,14 +12,14 @@ 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;
/**
* @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;
}
+8
View File
@@ -0,0 +1,8 @@
<?php
namespace Kiri\Server\Events;
class OnAfterRequest
{
}
+11
View File
@@ -2,7 +2,18 @@
namespace Kiri\Server\Events;
use Swoole\Server;
class OnAfterWorkerStart
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
}
+7 -1
View File
@@ -2,10 +2,16 @@
namespace Kiri\Server\Events;
use Swoole\Server;
class OnBeforeWorkerStart
{
public function __construct(public int $workerId)
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server,public int $workerId)
{
}
+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)
{
}
}
+4 -2
View File
@@ -2,6 +2,8 @@
namespace Kiri\Server\Events;
use Swoole\Coroutine\Http\Server as CHServer;
use Swoole\Coroutine\Server as CServer;
use Swoole\Server;
class OnShutdown
@@ -9,9 +11,9 @@ class OnShutdown
/**
* @param Server|null $server
* @param Server|CHServer|CServer|null $server
*/
public function __construct(public ?Server $server = null)
public function __construct(public Server|null|CHServer|CServer $server = null)
{
}
+7 -7
View File
@@ -11,13 +11,13 @@ class OnTaskerStart
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
}
+5 -5
View File
@@ -11,11 +11,11 @@ class OnWorkerExit
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
/**
* @param Server|null $server
* @param int $workerId
*/
public function __construct(public ?Server $server, public int $workerId)
{
}
+9 -7
View File
@@ -3,6 +3,8 @@
namespace Kiri\Server\Events;
use Swoole\Server;
use Swoole\Coroutine\Server as CServer;
use Swoole\Coroutine\Http\Server as CHServer;
/**
*
@@ -11,13 +13,13 @@ class OnWorkerStart
{
/**
* @param Server $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
{
}
/**
* @param Server|CHServer|CServer|null $server
* @param int $workerId
*/
public function __construct(public Server|null|CHServer|CServer $server, public int $workerId)
{
}
}
+4 -2
View File
@@ -2,6 +2,8 @@
namespace Kiri\Server\Events;
use Swoole\Coroutine\Http\Server as CHServer;
use Swoole\Coroutine\Server as CServer;
use Swoole\Server;
/**
@@ -12,10 +14,10 @@ class OnWorkerStop
/**
* @param Server $server
* @param Server|CHServer|CServer|null $server
* @param int $workerId
*/
public function __construct(public Server $server, public int $workerId)
public function __construct(public Server|null|CHServer|CServer $server, public int $workerId)
{
}
-37
View File
@@ -1,37 +0,0 @@
<?php
namespace Kiri\Server\Handler;
use Exception;
use Kiri\Annotation\Inject;
use Kiri\Events\EventDispatch;
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 Exception
*/
public function onPipeMessage(\Swoole\Server $server, int $src_worker_id, mixed $message)
{
if (is_string($message)) {
$message = unserialize($message);
}
if (!is_object($message) || !($message instanceof OnPipeMessageInterface)) {
return;
}
call_user_func([$message, 'process'], $server, $src_worker_id);
}
}
+40 -33
View File
@@ -2,14 +2,15 @@
namespace Kiri\Server\Handler;
use Kiri\Annotation\Inject;
use Kiri\Events\EventDispatch;
use Kiri\Exception\ConfigException;
use ReflectionException;
use Kiri;
use Kiri\Di\Inject\Container;
use Kiri\Error\StdoutLogger;
use Psr\Log\LoggerInterface;
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;
/**
@@ -18,39 +19,45 @@ use Kiri\Server\Events\OnStart;
*/
class OnServer extends Server
{
/**
* @param \Swoole\Server $server
* @throws ConfigException
* @throws ReflectionException
*/
public function onStart(\Swoole\Server $server)
{
$this->setProcessName(sprintf('start[%d].server', $server->master_pid));
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnStart($server));
}
/**
* @param \Swoole\Server $server
* @throws ReflectionException
*/
public function onBeforeShutdown(\Swoole\Server $server)
{
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnBeforeShutdown($server));
}
/**
* @var StdoutLogger
*/
#[Container(LoggerInterface::class)]
public StdoutLogger $logger;
/**
* @param \Swoole\Server $server
* @throws ReflectionException
*/
public function onShutdown(\Swoole\Server $server)
{
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnShutdown($server));
}
/**
* @param SServer $server
* @throws
*/
public function onStart(SServer $server): void
{
Kiri::setProcessName(sprintf('Master[%d]', $server->master_pid));
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));
}
}
+18 -20
View File
@@ -2,11 +2,8 @@
namespace Kiri\Server\Handler;
use Kiri\Annotation\Inject;
use Kiri\Events\EventDispatch;
use ReflectionException;
use Kiri;
use Kiri\Server\Abstracts\Server;
use Kiri\Exception\ConfigException;
use Kiri\Server\Events\OnManagerStart;
use Kiri\Server\Events\OnManagerStop;
@@ -19,26 +16,27 @@ class OnServerManager extends Server
{
/**
/**
* @param \Swoole\Server $server
* @throws ConfigException|ReflectionException
*/
public function onManagerStart(\Swoole\Server $server)
{
$this->setProcessName(sprintf('manger[%d].0', $server->manager_pid));
* @throws
*/
public function onManagerStart(\Swoole\Server $server): void
{
Kiri::setProcessName(sprintf('Manger[%d]', $server->manager_pid));
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnManagerStart($server));
}
event(new OnManagerStart($server));
}
/**
* @param \Swoole\Server $server
* @throws ReflectionException
*/
public function onManagerStop(\Swoole\Server $server)
{
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnManagerStop($server));
}
/**
* @param \Swoole\Server $server
* @return void
* @throws
*/
public function onManagerStop(\Swoole\Server $server): void
{
event(new OnManagerStop($server));
}
}
-38
View File
@@ -1,38 +0,0 @@
<?php
namespace Kiri\Server\Handler;
use Kiri\Annotation\Inject;
use Kiri\Events\EventDispatch;
use Kiri\Server\Events\OnAfterReload;
use Kiri\Server\Events\OnBeforeReload;
use Swoole\Server;
/**
*
*/
class OnServerReload
{
/**
* @param Server $server
* @throws \ReflectionException
*/
public function onBeforeReload(Server $server)
{
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnBeforeReload($server));
}
/**
* @param Server $server
* @throws \ReflectionException
*/
public function onAfterReload(Server $server)
{
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnAfterReload($server));
}
}
+83 -43
View File
@@ -2,88 +2,123 @@
namespace Kiri\Server\Handler;
use Exception;
use Kiri;
use Kiri\Abstracts\Config;
use Kiri\Annotation\Inject;
use Kiri\Abstracts\CoordinatorManager;
use Kiri\Coordinator;
use Kiri\Core\Help;
use Kiri\Di\Inject\Container;
use Kiri\Events\EventDispatch;
use Kiri\Message\Handler\Router;
use Kiri\Server\Events\OnAfterWorkerStart;
use Kiri\Server\Events\OnBeforeWorkerStart;
use Kiri\Server\Events\OnTaskerStart as OnTaskStart;
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 Psr\Http\Message\RequestInterface;
use Swoole\Server;
use Swoole\Timer;
use Throwable;
use function config;
/**
* Class OnServerWorker
* @package Server\Worker
*/
class OnServerWorker extends \Kiri\Server\Abstracts\Server
class OnServerWorker extends Kiri\Server\Abstracts\Server
{
public Router $collector;
/**
* @var EventDispatch
*/
#[Container(EventDispatch::class)]
public EventDispatch $dispatch;
/**
* @return void
*/
public function init()
public function init(): void
{
$this->collector = Kiri::getDi()->get(Router::class);
on(OnBeforeWorkerStart::class, [$this, 'onWorkerNameAlias']);
}
/**
* @param Server $server
* @param int $workerId
* @throws Exception
* @param OnBeforeWorkerStart $workerStart
* @return void
*/
public function onWorkerStart(Server $server, int $workerId)
public function onWorkerNameAlias(OnBeforeWorkerStart $workerStart): void
{
$dispatch = \Kiri::getDi()->get(EventDispatch::class);
$dispatch->dispatch(new OnBeforeWorkerStart($workerId));
set_env('environmental_workerId', $workerId);
if ($workerId < $server->setting['worker_num']) {
$this->setProcessName(sprintf('Worker[%d].%d', $server->worker_pid, $workerId));
$this->collector->scan_build_route();
$dispatch->dispatch(new OnWorkerStart($server, $workerId));
if ($workerStart->workerId < $workerStart->server->setting['worker_num']) {
$this->processName($workerStart->server, 'Worker');
set_env('environmental', Kiri::WORKER);
} else {
$dispatch->dispatch(new OnTaskStart($server, $workerId));
$this->setProcessName(sprintf('Tasker[%d].%d', $server->worker_pid, $workerId));
$this->processName($workerStart->server, 'Tasker');
set_env('environmental', Kiri::TASK);
}
$dispatch->dispatch(new OnAfterWorkerStart());
set_env('environmental_worker_id', $workerStart->workerId);
}
/**
* @param Server $server
* @param int $workerId
* @throws Exception
* @return void
* @throws
*/
public function onWorkerStop(Server $server, int $workerId)
public function onWorkerStart(Server $server, int $workerId): void
{
try {
$this->dispatch->dispatch(new OnBeforeWorkerStart($server, $workerId));
if ($workerId < $server->setting['worker_num']) {
CoordinatorManager::utility(Coordinator::WORKER_START)->wait();
$this->dispatch->dispatch(new OnWorkerStart($server, $workerId));
} else {
$this->dispatch->dispatch(new OnTaskerStart($server, $workerId));
}
} catch (Throwable $exception) {
\Kiri::getLogger()->json_log($exception);
} finally {
$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 . '[%d]', $server->worker_pid));
}
/**
* @param Server $server
* @param int $workerId
* @throws
*/
public function onWorkerStop(Server $server, int $workerId): void
{
event(new OnWorkerStop($server, $workerId));
Timer::clearAll();
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnWorkerStop($server, $workerId));
}
/**
* @param Server $server
* @param int $workerId
* @throws Exception
* @throws
*/
public function onWorkerExit(Server $server, int $workerId)
public function onWorkerExit(Server $server, int $workerId): void
{
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnWorkerExit($server, $workerId));
event(new OnWorkerExit($server, $workerId));
}
@@ -93,35 +128,40 @@ class OnServerWorker extends \Kiri\Server\Abstracts\Server
* @param int $worker_pid
* @param int $exit_code
* @param int $signal
* @throws Exception
* @throws
*/
public function onWorkerError(Server $server, int $worker_id, int $worker_pid, int $exit_code, int $signal)
public function onWorkerError(Server $server, int $worker_id, int $worker_pid, int $exit_code, int $signal): void
{
\Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnWorkerError($server, $worker_id, $worker_pid, $exit_code, $signal));
event(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);
debug_print_backtrace(DEBUG_BACKTRACE_PROVIDE_OBJECT);
/** @var RequestInterface $context */
$context = Kiri\Di\Context::get(RequestInterface::class);
if (is_null($context)) {
$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));
} else {
$message = sprintf('Worker#%d::%d error stop. signal %d, exit_code %d, msg %s, method: %s, path: %s, query: %s', $worker_id, $worker_pid, $signal, $exit_code, swoole_strerror(swoole_last_error(), $signal),
$context->getMethod(), $context->getUri()->getPath(), $context->getUri()->getQuery());
}
$this->getLogger()->println($message);
$this->system_mail($message);
}
/**
* @param $messageContent
* @throws Exception
* @throws
*/
protected function system_mail($messageContent)
protected function system_mail($messageContent): void
{
try {
$email = Config::get('email', ['enable' => false]);
if (!empty($email) && ($email['enable'] ?? false) == true) {
$email = config('email', ['enable' => false]);
if (!empty($email) && ($email['enable'] ?? false)) {
Help::sendEmail($email, 'Service Error', $messageContent);
}
} catch (\Throwable $e) {
error($e, 'email');
} catch (Throwable $e) {
\Kiri::getLogger()->json_log($e);
}
}
-125
View File
@@ -1,125 +0,0 @@
<?php
namespace Kiri\Server;
use Kiri;
use Kiri\Abstracts\Config;
use Kiri\Annotation\Inject;
use Kiri\Context;
use Kiri\Events\EventDispatch;
use Kiri\Exception\ConfigException;
use Kiri\Server\Abstracts\BaseProcess;
use Kiri\Server\Broadcast\Message;
use Kiri\Server\Contract\OnProcessInterface;
use Kiri\Server\Events\OnProcessStart;
use Psr\Log\LoggerInterface;
use Swoole\Coroutine;
use Swoole\Process;
class ProcessManager
{
/** @var array<string, Process> */
private array $_process = [];
#[Inject(LoggerInterface::class)]
public LoggerInterface $logger;
/**
* @param string|OnProcessInterface|BaseProcess $customProcess
* @return void
* @throws ConfigException
*/
public function add(string|OnProcessInterface|BaseProcess $customProcess)
{
$server = Kiri::getDi()->get(SwooleServerInterface::class);
if (is_string($customProcess)) {
$customProcess = Kiri::getDi()->get($customProcess);
}
$system = sprintf('[%s].process', Config::get('id', 'system-service'));
$this->logger->debug($system . ' ' . $customProcess->getName() . ' start.');
$process = $this->parse($customProcess, $system);
if (Context::inCoroutine()) {
Coroutine::create(function () use ($process) {
$process->start();
});
} else {
$server->addProcess($process = $this->parse($customProcess, $system));
}
$this->_process[$customProcess->getName()] = $process;
}
/**
* @param $customProcess
* @param $system
* @return Process
*/
private function parse($customProcess, $system): Process
{
return new Process(function (Process $process) use ($customProcess, $system) {
if (Kiri::getPlatform()->isLinux()) {
$process->name($system . '(' . $customProcess->getName() . ')');
}
Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnProcessStart());
set_env('environmental', Kiri::PROCESS);
$channel = Coroutine::create(function () use ($process, $customProcess) {
while (!$customProcess->isStop()) {
$message = $process->read();
if (!empty($message)) {
$message = unserialize($message);
}
if (is_null($message)) {
continue;
}
$customProcess->onBroadcast($message);
}
});
Context::setContext('waite:process:message', $channel);
$customProcess->onSigterm()->process($process);
},
$customProcess->getRedirectStdinAndStdout(),
$customProcess->getPipeType(),
$customProcess->isEnableCoroutine()
);
}
/**
* @param array $processes
* @return void
* @throws ConfigException
*/
public function batch(array $processes)
{
foreach ($processes as $process) {
$this->add($process);
}
}
/**
* @param string $message
* @param string $name
* @return void
*/
public function push(string $message, string $name = '')
{
$processes = $this->_process;
if (!empty($this->_process[$name])) {
$processes = [$this->_process[$name]];
}
foreach ($processes as $process) {
$process->write(serialize(new Message($message)));
}
}
}
+172
View File
@@ -0,0 +1,172 @@
<?php
namespace Kiri\Server\Processes;
use Swoole\Coroutine;
use Swoole\Process;
use const SIGHUP;
use const SIGTERM;
/**
*
*/
abstract class AbstractProcess implements OnProcessInterface
{
private bool $stop = false;
public Process $process;
/**
* @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 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 void
*/
abstract public function onSigterm(): void;
/**
* @param Process $process
* @return AbstractProcess
*/
public function onShutdown(Process $process): static
{
$this->process = $process;
if ($this->enable_coroutine) {
$array['enable_deadlock_check'] = false;
$array['deadlock_check_disable_trace'] = false;
$array['exit_condition'] = [$this, 'exit_condition'];
Coroutine::set($array);
Coroutine::create(fn() => $this->coroutineWaitSignal());
} else {
$process::signal(SIGTERM | SIGINT | SIGUSR1, [$this, 'pointWaitSignal']);
}
return $this;
}
public function exit_condition(): bool
{
return Coroutine::stats()['coroutine_num'] === 0;
}
/**
* @param $signal
* @return void
*/
public function pointWaitSignal($signal): void
{
$this->stop = true;
$this->onSigterm();
}
/**
* @return void
*/
public function coroutineWaitSignal(): void
{
$value = Coroutine::waitSignal(SIGTERM);
if ($value) {
$this->stop = true;
}
$this->onSigterm();
}
}
+25
View File
@@ -0,0 +1,25 @@
<?php
namespace Kiri\Server\Processes;
use Swoole\Process;
/**
* Interface AbstractProcess
* @package Contract
*/
interface OnProcessInterface
{
/**
* @param ?Process $process
*/
public function process(?Process $process): void;
}
+69
View File
@@ -0,0 +1,69 @@
<?php
namespace Kiri\Server\Processes;
use Exception;
use Kiri;
use Swoole\Process;
trait TraitProcess
{
/**
* @var array
*/
private array $_process = [];
/**
* @param string|array|AbstractProcess $class
* @return void
* @throws
*/
public function addProcess(string|array|AbstractProcess $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('AbstractProcess(' . $name->getName() . ') is exists.');
}
$this->_process[$name->getName()] = $this->genProcess($name);
}
}
/**
* @param AbstractProcess $name
* @return Process
*/
private function genProcess(AbstractProcess $name): Process
{
return new Process(function (Process $process) use ($name) {
$process->name('[' . \config('site.id', 'system-service') . '].' . $name->getName() . '[' . $process->pid . ']');
$name->onShutdown($process)->process($process);
},
$name->getRedirectStdinAndStdout(),
$name->getPipeType(),
$name->isEnableCoroutine());
}
/**
* @param string $name
* @return AbstractProcess|null
*/
public function getProcess(string $name): ?Process
{
return $this->_process[$name] ?? null;
}
/**
* @return array
*/
public function getProcesses(): array
{
return $this->_process;
}
}
-157
View File
@@ -1,157 +0,0 @@
<?php
namespace Kiri\Server;
use Exception;
use JetBrains\PhpStorm\Pure;
use Kiri;
use Kiri\Abstracts\Config;
use Kiri\Annotation\Inject;
use Kiri\Events\EventDispatch;
use Kiri\Exception\ConfigException;
use Kiri\Message\Handler\Abstracts\HttpService;
use Kiri\Server\Events\OnShutdown;
use Psr\Container\ContainerExceptionInterface;
use Kiri\Server\Events\OnServerBeforeStart;
use Psr\Container\NotFoundExceptionInterface;
use ReflectionException;
use Swoole\Coroutine;
defined('PID_PATH') or define('PID_PATH', APP_PATH . 'storage/server.pid');
/**
* Class Server
* @package Http
*/
class Server extends HttpService
{
private array $process = [
];
private mixed $daemon = 0;
/**
* @var State
*/
#[Inject(State::class)]
public State $state;
public ServerManager $manager;
/**
*
*/
public function init()
{
$this->manager = Kiri::getContainer()->get(ServerManager::class);
$enable_coroutine = Config::get('servers.settings.enable_coroutine', false);
Config::set('servers.settings.enable_coroutine', true);
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 $process
*/
public function addProcess($process)
{
$this->process[] = $process;
}
/**
* @return string
* @throws ConfigException
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
* @throws ReflectionException
* @throws Exception
*/
public function start(): mixed
{
$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', []));
$this->getContainer()->get(ProcessManager::class)->batch($processes);
$this->getEventDispatch()->dispatch(new OnServerBeforeStart());
return $this->manager->getServer()->start();
}
/**
* @return void
* @throws ConfigException
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
* @throws ReflectionException
* @throws Exception
*/
public function shutdown()
{
$configs = Config::get('server', [], true);
foreach ($this->manager->sortService($configs['ports'] ?? []) as $config) {
$this->state->exit($config['port']);
}
$this->getContainer()->get(EventDispatch::class)->dispatch(new OnShutdown());
}
/**
* @return bool
* @throws ConfigException
* @throws Exception
*/
public function isRunner(): bool
{
return $this->state->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();
}
}
+118 -48
View File
@@ -5,19 +5,26 @@ namespace Kiri\Server;
use Exception;
use Kiri\Abstracts\Config;
use Kiri\Events\EventDispatch;
use Kiri\Exception\ConfigException;
use Kiri;
use Kiri\Annotation\Inject;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\NotFoundExceptionInterface;
use Swoole\Coroutine;
use Kiri\Server\Abstracts\FileWatcher;
use Kiri\Server\Events\OnWorkerStart;
use Kiri\Events\EventProvider;
use Kiri\Events\EventDispatch;
use Kiri\Router\Router;
use Kiri\Server\Abstracts\AsyncServer;
use Kiri\Server\Events\OnShutdown;
use Psr\Container\ContainerInterface;
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;
use Kiri\Server\Abstracts\HotReload;
use Kiri\Di\Inject\Container;
defined('ROUTER_TYPE_HTTP') or define('ROUTER_TYPE_HTTP', 'http');
defined('PID_PATH') or define('PID_PATH', APP_PATH . 'storage/server.pid');
/**
* Class Command
@@ -27,51 +34,114 @@ class ServerCommand extends Command
{
const ACTIONS = ['start', 'stop', 'restart'];
#[Container(State::class)]
public State $state;
/**
*
*/
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');
}
/**
* @var ContainerInterface
*/
#[Container(ContainerInterface::class)]
public ContainerInterface $container;
/**
* @param InputInterface $input
* @param OutputInterface $output
* @return int
* @throws ConfigException
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
* @throws \ReflectionException
* @throws Exception
*/
public function execute(InputInterface $input, OutputInterface $output): int
{
$manager = Kiri::app()->getServer();
$manager->setDaemon((int)!is_null($input->getOption('daemon')));
/**
* @var AsyncServer
*/
#[Container(AsyncServer::class)]
public AsyncServer $asyncServer;
$action = $input->getArgument('action');
if (is_null($action)) {
throw new Exception('I don\'t know what I want to do.');
}
if (!in_array($action, self::ACTIONS)) {
throw new Exception('I don\'t know what I want to do.');
}
if ($action == 'restart' || $action == 'stop') {
$manager->shutdown();
if ($action == 'stop') {
return 1;
}
}
return (int)$manager->start();
}
/**
* @var EventDispatch
*/
#[Container(EventDispatch::class)]
public EventDispatch $eventDispatch;
/**
* @var EventProvider
*/
#[Container(EventProvider::class)]
public EventProvider $eventProvider;
/**
* @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('servers.server', []);
$instances = $this->asyncServer->sortService($configs['ports'] ?? []);
foreach ($instances as $config) {
$this->state->exit($config->port);
}
$this->eventDispatch->dispatch(new OnShutdown());
return 1;
}
/**
* @param InputInterface $input
* @return int
* @throws
*/
protected function start(InputInterface $input): int
{
$this->asyncServer->addProcess(config('process', []));
if (\config('servers.reload.hot', false) === true) {
$this->asyncServer->addProcess([FileWatcher::class]);
}
// Master 进程在 fork 前完成首次扫描,Worker 启动时不再重复全量扫描
// 避免每个 Worker 独立执行 opcache_compile_file + invalidateClasses 造成 OOM
di(Router::class)->scan_build_route();
$this->asyncServer->initCoreServers(config('servers.server', []), (int)$input->getOption('daemon'));
$this->asyncServer->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;
}
-309
View File
@@ -1,309 +0,0 @@
<?php
namespace Kiri\Server;
use Exception;
use Kiri\Abstracts\Component;
use Kiri\Abstracts\Config;
use Kiri\Annotation\Inject;
use Kiri\Error\Logger;
use Kiri\Exception\ConfigException;
use Kiri\Server\Contract\OnCloseInterface;
use Kiri\Server\Contract\OnConnectInterface;
use Kiri\Server\Contract\OnDisconnectInterface;
use Kiri\Server\Contract\OnHandshakeInterface;
use Kiri\Server\Contract\OnMessageInterface;
use Kiri\Server\Contract\OnPacketInterface;
use Kiri\Server\Contract\OnReceiveInterface;
use Kiri\Server\Handler\OnPipeMessage;
use Kiri\Server\Handler\OnServer;
use Kiri\Server\Handler\OnServerWorker;
use Kiri\Task\TaskManager;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\NotFoundExceptionInterface;
use Swoole\Http\Server as HServer;
use Swoole\Server;
use Swoole\Server\Port;
use Swoole\WebSocket\Server as WServer;
/**
* Class OnServerManager
* @package Http\Service
*/
class ServerManager extends Component
{
use TraitServer;
/** @var string */
public string $host = '';
public int $port = 0;
/**
* @var Logger
*/
#[Inject(Logger::class)]
public Logger $logger;
/** @var array<string,Port> */
public array $ports = [];
public int $mode = SWOOLE_TCP;
private Server|null $server = null;
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::START => [OnServer::class, 'onStart'],
Constant::SHUTDOWN => [OnServer::class, 'onShutdown'],
];
/**
* @var array|string[]
*/
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 ConfigException
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
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);
}
}
/**
* @param $configs
* @param int $daemon
* @throws ConfigException
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
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->bindPipeMessage();
}
/**
* @return void
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
public function bindPipeMessage(): void
{
$pipeMessage = $this->getContainer()->get(OnPipeMessage::class);
$this->server->on(Constant::PIPE_MESSAGE, [$pipeMessage, 'onPipeMessage']);
if (!isset($this->server->setting['task_worker_num']) || $this->server->setting['task_worker_num'] < 1) {
return;
}
$this->getContainer()->get(TaskManager::class)->taskListener($this->server);
}
/**
* @return array
*/
public function getSetting(): array
{
return $this->server->setting;
}
/**
* @param ServerManager $context
* @param array $config
* @param int $daemon
* @throws ConfigException
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
* @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 ConfigException
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
* @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 ConfigException
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
* @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']));
$this->getContainer()->setBindings(SwooleServerInterface::class, $this->server);
$id = Config::get('id', 'system-service');
$this->logger->debug(sprintf('[%s].' . $type . ' service %s::%d start', $id, $host, $port));
$this->addDefaultListener($settings);
}
/**
* @param array $settings
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
private function addDefaultListener(array $settings): void
{
$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
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
private function addServiceEvents(array $events, Server|Port $server)
{
foreach ($events as $name => $event) {
if (is_array($event) && is_string($event[0])) {
$event[0] = $this->getContainer()->get($event[0]);
}
$server->on($name, $event);
}
}
/**
*
*/
public function start()
{
$this->server->start();
}
}
+10 -14
View File
@@ -4,10 +4,8 @@ declare(strict_types=1);
namespace Kiri\Server;
use Exception;
use Kiri\Abstracts\Providers;
use Kiri\Application;
use Kiri;
use Symfony\Component\Console\Application;
/**
* Class DatabasesProviders
@@ -17,16 +15,14 @@ class ServerProviders extends Providers
{
/**
* @param Application $application
* @throws Exception
*/
public function onImport(Application $application)
{
$container = Kiri::getDi();
/**
* @throws
*/
public function onImport(): void
{
$server = $this->container->get(ServerCommand::class);
$console = $container->get(\Symfony\Component\Console\Application::class);
$console->add($container->get(ServerCommand::class));
}
$console = $this->container->get(Application::class);
$console->addCommand($server);
}
}
+39 -36
View File
@@ -2,55 +2,58 @@
namespace Kiri\Server;
use Exception;
use Kiri\Abstracts\Component;
use Kiri\Abstracts\Config;
use Kiri\Server\Abstracts\TraitServer;
use Swoole\Process;
use function config;
class State extends Component
{
use TraitServer;
use TraitServer;
public array $servers = [];
public array $servers = [];
public function init()
{
$this->servers = Config::get('server.ports');
}
/**
* @return void
*/
public function init(): void
{
$this->servers = config('servers.server.ports');
}
/**
* @return bool
* @throws Exception
*/
public function isRunner(): bool
{
$ports = $this->sortService($this->servers);
foreach ($ports as $config) {
if (checkPortIsAlready($config['port'])) {
return true;
}
}
return false;
}
/**
* @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 Exception
*/
public function exit($port)
{
if (!($pid = checkPortIsAlready($port))) {
return;
}
while (checkPortIsAlready($port)) {
Process::kill($pid, SIGTERM);
usleep(300);
}
}
/**
* @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 Kiri\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(public int $task_id, public mixed $data)
{
}
}
+17
View File
@@ -0,0 +1,17 @@
<?php
namespace Kiri\Server\Task;
interface OnTaskInterface
{
/**
* @param int $task_id
* @param int $src_worker_id
* @return mixed
*/
public function process(int $task_id, int $src_worker_id): mixed;
}
+94
View File
@@ -0,0 +1,94 @@
<?php
namespace Kiri\Server\Task;
use Kiri;
use Kiri\Router\Base\ExceptionHandlerDispatcher;
use Kiri\Router\Interface\ExceptionHandlerInterface;
use Kiri\Server\Constant;
use Swoole\Server;
/**
*
*/
class Task
{
public ExceptionHandlerInterface $exception;
/**
*
*/
public function __construct()
{
$exception = \config('servers.task.exception');
if (!in_array(ExceptionHandlerInterface::class, class_implements($exception))) {
$exception = ExceptionHandlerDispatcher::class;
}
$this->exception = Kiri::getDi()->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 void
* @throws
*/
public function onTask(Server $server, int $task_id, int $src_worker_id, mixed $data): void
{
try {
[$handler, $params] = [$data[0], $data[1]];
$handler = Kiri::getDi()->make($handler, $params);
if (!($handler instanceof OnTaskInterface)) {
throw new \Exception('Task process must implements ' . OnTaskInterface::class);
}
$response = call_user_func([$handler, 'process'], $task_id, $src_worker_id);
} catch (\Throwable $throwable) {
\Kiri::getLogger()->json_log($throwable, ['task_id' => $task_id, 'src_worker_id' => $src_worker_id, 'data' => $data]);
$response = throwable($throwable);
} finally {
$server->finish($response);
}
}
}
+76
View File
@@ -0,0 +1,76 @@
<?php
namespace Kiri\Server\Task;
use Exception;
use Kiri\Server\ServerInterface;
class TaskExecute implements TaskInterface
{
/**
* @param string $handler
* @param mixed $data
* @param int $dstWorkerId
* @param callable|null $finishFinishCallback
* @return void
* @throws
*/
public function task(string $handler, mixed $data, int $dstWorkerId = -1, ?callable $finishFinishCallback = null): void
{
$array = class_implements($handler, true);
if (!in_array(OnTaskInterface::class, $array, true)) {
throw new Exception('Task is not implement OnTaskInterface');
}
$server = \Kiri::getDi()->get(ServerInterface::class);
$server->task([$handler, $data], $dstWorkerId, $finishFinishCallback);
}
/**
* @param string $handler
* @param mixed $data
* @param float $timeout
* @param int $dstWorkerId
* @return mixed
* @throws
*/
public function taskWait(string $handler, mixed $data, float $timeout = 0.5, int $dstWorkerId = -1): mixed
{
$array = class_implements($handler, true);
if (!in_array(OnTaskInterface::class, $array, true)) {
throw new Exception('Task is not implement OnTaskInterface');
}
$server = \Kiri::getDi()->get(ServerInterface::class);
return $server->taskwait([$handler, $data], $timeout, $dstWorkerId);
}
/**
* @param array $tasks
* @param float $timeout
* @return false|array
*/
public function taskCo(array $tasks, float $timeout = 0.5): false|array
{
$server = \Kiri::getDi()->get(ServerInterface::class);
return $server->taskCo($tasks, $timeout);
}
/**
* @param array $tasks
* @param float $timeout
* @return false|array
*/
public function taskWaitMulti(array $tasks, float $timeout = 0.5): false|array
{
$server = \Kiri::getDi()->get(ServerInterface::class);
return $server->taskWaitMulti($tasks, $timeout);
}
}
+44
View File
@@ -0,0 +1,44 @@
<?php
namespace Kiri\Server\Task;
interface TaskInterface
{
/**
* @param string $handler
* @param mixed $data
* @param int $dstWorkerId
* @param callable|null $finishFinishCallback
* @return void
*/
public function task(string $handler, mixed $data, int $dstWorkerId = -1, ?callable $finishFinishCallback = null): void;
/**
* @param string $handler
* @param mixed $data
* @param float $timeout
* @param int $dstWorkerId
* @return mixed
*/
public function taskWait(string $handler, mixed $data, float $timeout = 0.5, int $dstWorkerId = -1): mixed;
/**
* @param array $tasks
* @param float $timeout
* @return false|array
*/
public function taskCo(array $tasks, float $timeout = 0.5): false|array;
/**
* @param array $tasks
* @param float $timeout
* @return false|array
*/
public function taskWaitMulti(array $tasks, float $timeout = 0.5): false|array;
}
+24
View File
@@ -0,0 +1,24 @@
<?php
namespace Kiri\Server\Task;
class TestTask implements OnTaskInterface
{
public function __construct($user, $friend)
{
}
/**
* @param int $task_id
* @param int $src_worker_id
* @return int
*/
public function process(int $task_id, int $src_worker_id): int
{
return 0;
}
}
-32
View File
@@ -1,32 +0,0 @@
<?php
namespace Kiri\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": "~v2.0"
},
"autoload": {
"psr-4": {
"Kiri\\Server\\": "./"
}
"name": "game-worker/kiri-http-server",
"description": "http-server",
"license": "MIT",
"authors": [
{
"name": "向林",
"email": "as2252258@163.com"
}
],
"require": {
"php": ">=8.5",
"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\\": "./"
}
}
}