Compare commits
1600 Commits
v1.33.3
...
316537eb4d
| Author | SHA1 | Date | |
|---|---|---|---|
| 316537eb4d | |||
| b2c421600c | |||
| 787f6ef528 | |||
| 8578547467 | |||
| 51ab6d6da1 | |||
| 3a8b5de8f7 | |||
| faf08f6513 | |||
| 06c69d23be | |||
| 1bcadd5f8e | |||
| 524195d729 | |||
| 0c25d277ef | |||
| 27b0348e1d | |||
| ea5aa68769 | |||
| 99fefb168a | |||
| 49457505cd | |||
| bfc948a9b4 | |||
| c407206627 | |||
| 39d3bf97d1 | |||
| 79be392775 | |||
| be4c6b8e16 | |||
| c8e193e70d | |||
| 35859ffc3f | |||
| 0d81ada5a8 | |||
| b68cf0fb45 | |||
| 9ed2078e92 | |||
| 1f002159e2 | |||
| 5bc690fcd9 | |||
| bab9adce24 | |||
| e47eddaa85 | |||
| 8ef1f2e395 | |||
| 541131bbc6 | |||
| 7626eecbf6 | |||
| 49afa75929 | |||
| 5c5265ede2 | |||
| 42d61d8089 | |||
| 01eb50078e | |||
| eef021f472 | |||
| 6f3fd785e7 | |||
| 7cd8a645a8 | |||
| 9671348dc1 | |||
| 7a3e68d656 | |||
| 42c7ec2f75 | |||
| 32c3ce5c98 | |||
| e55a3a82fc | |||
| 305da86f97 | |||
| c23d1d11b5 | |||
| a3cabd5f36 | |||
| 66ac4716f2 | |||
| 3cd1aaeb03 | |||
| 4eb940ffe7 | |||
| 61800b23e2 | |||
| 0283662931 | |||
| d125eb56b3 | |||
| 956d68695c | |||
| 83d81b64b3 | |||
| a4ea82c04b | |||
| 8387fe0d5b | |||
| cfd5b388f1 | |||
| 4ee6e38a94 | |||
| 3f87752d1f | |||
| b91548eef4 | |||
| 1195417b97 | |||
| 67f347197e | |||
| 8c2dfa9140 | |||
| a3fbfe0bff | |||
| 99db1b1cc3 | |||
| 638a7f0ab4 | |||
| 806a69fef3 | |||
| 8ba2e9e34c | |||
| e7e54bc19e | |||
| 9fb980599f | |||
| 28dfef985c | |||
| 1e416b9f8a | |||
| 784bcb0aa5 | |||
| 9642df2d9d | |||
| 37340838b6 | |||
| d1a8dd7817 | |||
| 8919a3937a | |||
| 5032030f8d | |||
| b30cb5d7dc | |||
| 7113c4622b | |||
| bd8caff0b7 | |||
| 519bf3184a | |||
| 79c77ce3a3 | |||
| 2f40f795ee | |||
| b16f92314b | |||
| ad22244388 | |||
| 02f89a9c9d | |||
| d286c040a5 | |||
| 99f5b8ebc1 | |||
| 9ac33f9b9b | |||
| 6ab1fcaf89 | |||
| 2e3d0cc57c | |||
| 1e44115461 | |||
| 8d57063e9d | |||
| 104d646c7c | |||
| 9ddbf79d9e | |||
| a9ec4c5c28 | |||
| 914d860197 | |||
| 23b3e5c731 | |||
| cdf04c2402 | |||
| 3535e44337 | |||
| 0b245d3885 | |||
| 4fda6cbcde | |||
| 2bbba897ec | |||
| 0cfb94b0ba | |||
| 3f7ac93932 | |||
| 96c36b4f6a | |||
| febd6d32cf | |||
| cbd8699801 | |||
| 1ee1d61c74 | |||
| 74400aacc6 | |||
| 9f55c3605a | |||
| 8d61e8179f | |||
| f250889c3e | |||
| 00f67d86d6 | |||
| 7b8b9cfd2b | |||
| 5e7a67834b | |||
| 3c85602ab1 | |||
| 66d0d0e213 | |||
| 1f68faddb9 | |||
| db06f06c96 | |||
| 5b580d2a17 | |||
| 083dd7d1a3 | |||
| 03bd4755ce | |||
| 79e973e9c8 | |||
| 29d37075dd | |||
| f311bac580 | |||
| beb7a4c992 | |||
| 4d86fb319b | |||
| 5ea4f46de7 | |||
| 1d8d5251ae | |||
| 54c8217808 | |||
| ba623903e0 | |||
| 907af3ae18 | |||
| 24ae8a6b66 | |||
| 1646a5cdd2 | |||
| 814f17d10b | |||
| 40fe105903 | |||
| 42a347d8b1 | |||
| 5450e5dac4 | |||
| 1368259a1e | |||
| 81a495f267 | |||
| 693a4a6633 | |||
| 82786c580a | |||
| e19743f705 | |||
| 9166a57930 | |||
| 8d8304e859 | |||
| 6ddc23e2aa | |||
| 2fc491015e | |||
| bb0afe1fa7 | |||
| eb46f8c776 | |||
| bd511f97cb | |||
| 560bf40e4b | |||
| 4f4652c1cd | |||
| 60e13c2a1d | |||
| 1fe0dc4d16 | |||
| 181a1e3c0a | |||
| 6bba771856 | |||
| 921f1f42fb | |||
| eeb1f27fa4 | |||
| 9ce21ad152 | |||
| c036929cfe | |||
| 21591a3a89 | |||
| a2e9a41a7e | |||
| 0902349130 | |||
| f900db8e10 | |||
| 0fa9b344e0 | |||
| f48ef3d17b | |||
| 40801d0a06 | |||
| c6ccf1cf21 | |||
| d311992983 | |||
| b4babbe2c7 | |||
| 0719f4c99e | |||
| eb5de15033 | |||
| b229486d3b | |||
| 33b8d3e219 | |||
| 230256793f | |||
| 540ef96745 | |||
| 1baf30a671 | |||
| 5e93840e48 | |||
| 73a5908039 | |||
| 8429148273 | |||
| 17f40e2180 | |||
| 0345b12379 | |||
| 163b0de874 | |||
| 1661caed05 | |||
| 1089aeab9e | |||
| 1a0d3eeb1b | |||
| c27636529d | |||
| ac85488245 | |||
| 433e98b645 | |||
| 873654669e | |||
| 8b96f218d5 | |||
| 52cbff0e15 | |||
| 32de8d9ccb | |||
| e06da886f7 | |||
| cdd5ad3a8e | |||
| 9bee0e460b | |||
| 60c8ace443 | |||
| 933aaeaf25 | |||
| ca43c77525 | |||
| b204182c13 | |||
| dfb4165a12 | |||
| 26a54cd228 | |||
| e229f14121 | |||
| ee6cdfb391 | |||
| 58354e563c | |||
| 48f1bf0918 | |||
| 0ffe1f2cef | |||
| 8fe1adacf3 | |||
| 30245c5d8a | |||
| 67a5225bde | |||
| 58c3d7087b | |||
| c408687af7 | |||
| 33b284afc0 | |||
| 78004bdfb5 | |||
| 640950d4c8 | |||
| 998de0f9a0 | |||
| ce6e515309 | |||
| e054c8fc55 | |||
| 9fa1c2eb3e | |||
| 64a314c19e | |||
| 40be42406c | |||
| bca0eefc83 | |||
| 662ca19f8f | |||
| 65f9d482f3 | |||
| 7058d5cb93 | |||
| f7b13c69e9 | |||
| 9f21b1a097 | |||
| 53983002b6 | |||
| d3c0914ac1 | |||
| 7eb9694221 | |||
| b8f0d37420 | |||
| 1d5b1c239c | |||
| d1ebc08478 | |||
| 5074a91669 | |||
| 40e56c4040 | |||
| 3b0ed9310a | |||
| f92dc6a1ad | |||
| 0040b76a19 | |||
| 6c6fbabf14 | |||
| 8c2d868093 | |||
| 96c9e74c6f | |||
| d947437c10 | |||
| 83df29d832 | |||
| 607afe864a | |||
| a97cee84f3 | |||
| ad64384891 | |||
| f75c73d739 | |||
| 418bcddc95 | |||
| 61192b998a | |||
| 5ea2b09dc3 | |||
| 5bfc2c4a9b | |||
| 8ec47c3894 | |||
| f4423638a2 | |||
| 7b3444308b | |||
| 5ec9916817 | |||
| be1a70299f | |||
| 8685aa371a | |||
| 0224faa184 | |||
| 8546e326cf | |||
| 9956fd2f04 | |||
| 4f669ca82f | |||
| 1cd3881aa8 | |||
| e634513f7b | |||
| 7b6cde6ae3 | |||
| 18146fdf9e | |||
| 2c80c35b21 | |||
| 54b73769b8 | |||
| f7983ee4d9 | |||
| 9eace86aee | |||
| 2fbb58eb2b | |||
| 187d04e3a1 | |||
| d5d7d73440 | |||
| b747e281b7 | |||
| e024d50476 | |||
| a6ba48c075 | |||
| e19375387d | |||
| a9f68187d4 | |||
| 4d754fa78d | |||
| 6d07ab2bc5 | |||
| a60b00c440 | |||
| d0f3f303b6 | |||
| 4fc8acce8c | |||
| 0797a4f99d | |||
| db453c8038 | |||
| c776c34cfd | |||
| 170b39fde6 | |||
| fc27a66825 | |||
| 06b49c140e | |||
| 3ab45c91e1 | |||
| 6660161cec | |||
| 8c6e207008 | |||
| 4180e3c540 | |||
| a218cd0ffb | |||
| 31c8de5bbe | |||
| faac7f365f | |||
| 8cc4332b10 | |||
| abd30da102 | |||
| f4701ff72a | |||
| 393d7885a1 | |||
| b4c6d3c975 | |||
| 3ce440a28d | |||
| cb4ab6a99f | |||
| 6c39d7b1ee | |||
| 840bd52671 | |||
| f3d70c7ea1 | |||
| 22b8528ee1 | |||
| 900e302df7 | |||
| be03d8e137 | |||
| d338a9639a | |||
| 235972f3da | |||
| ae822881e7 | |||
| 26f75c71ba | |||
| ad2aa2eff5 | |||
| 52689049ae | |||
| 1a29541140 | |||
| ece17eecef | |||
| 2b353094eb | |||
| 70305aa501 | |||
| 22dc504ad0 | |||
| ef9402d403 | |||
| f7e29532f7 | |||
| 60770683b6 | |||
| 051bbbc64f | |||
| 206eef964c | |||
| b8b7adff17 | |||
| 42c68d362e | |||
| 1df3967889 | |||
| dae87e26a3 | |||
| f193341eae | |||
| d21a042ad8 | |||
| b16b9e813d | |||
| a79fe1f350 | |||
| 14f99875fb | |||
| 2d2890b34f | |||
| 66d8cafd76 | |||
| 468ccbf2b7 | |||
| 6b6668f73b | |||
| 5eda05f007 | |||
| 1d6a8bd851 | |||
| e500af1ed4 | |||
| 7ee39fd4ed | |||
| 9ba6c83821 | |||
| 4243622414 | |||
| 8374a4f5bf | |||
| 5d851141cb | |||
| 26ac081182 | |||
| 64e0d9a4d5 | |||
| 269a6cad80 | |||
| 806ebdb8a8 | |||
| ed0016fd2b | |||
| a3fb24993d | |||
| 9c26598831 | |||
| 7634f153b7 | |||
| cabc4da3ac | |||
| 6419539305 | |||
| 97e1178525 | |||
| 04faf12c14 | |||
| e7aa79cc9f | |||
| f862e3f37f | |||
| ce051af5bb | |||
| dcb9f1c840 | |||
| 83d0fe9f09 | |||
| 7789ba4d36 | |||
| 80101b04e2 | |||
| 310d4d1b57 | |||
| 126c0c6ad6 | |||
| 6a43b44087 | |||
| 6772b32609 | |||
| db2f0b8c5c | |||
| 1376004197 | |||
| 1625989c48 | |||
| f7863bd686 | |||
| 883565905a | |||
| 786780ce9b | |||
| d2e9fed62d | |||
| 0e5a4fb098 | |||
| f223f042de | |||
| cd413825ed | |||
| a851c272cd | |||
| 412077b418 | |||
| 4df6f8a50e | |||
| 44bf4b1cc1 | |||
| 136e8dd7c5 | |||
| 024b2b04a4 | |||
| 5bbf210394 | |||
| 039c62b09b | |||
| 07f0aa45ef | |||
| de11f44309 | |||
| acee96ef17 | |||
| 4ed49f9dfa | |||
| f68b585f8f | |||
| 13ddc979ec | |||
| b0b7ac3efb | |||
| 62f8525dd5 | |||
| 1347355cb1 | |||
| f847c4a414 | |||
| 776fa924e3 | |||
| 8872466968 | |||
| b620038d98 | |||
| a248367b15 | |||
| c159ec4a9a | |||
| 5359a7670f | |||
| 7e1c7a6de2 | |||
| 91e19bbdd3 | |||
| e61daaee2d | |||
| 8caab1fd92 | |||
| cd944882c3 | |||
| 888d9591fe | |||
| 833808c5de | |||
| d731956b06 | |||
| 40449ae4de | |||
| 44ad61f004 | |||
| 74865d53f8 | |||
| 373415261e | |||
| d0f653da9a | |||
| cbb8319cfa | |||
| 0e467a6024 | |||
| e505916525 | |||
| 31f09ab117 | |||
| 09e5e0f9b3 | |||
| 773cada57a | |||
| 403947ed6d | |||
| d9d08a725c | |||
| e2ed75af94 | |||
| dd19afce92 | |||
| 5b5deac7d9 | |||
| 3f3ee3456e | |||
| 3e2f2fc02e | |||
| c5a3003cf7 | |||
| 4c6dcddf11 | |||
| b314e500cd | |||
| b83e6ad13f | |||
| fee401cfdf | |||
| fa14f62198 | |||
| 5526665494 | |||
| 6249af996a | |||
| e51a1b365e | |||
| f53f00d126 | |||
| ab8fbaf21d | |||
| 63d8bcf882 | |||
| e4e16bc6a6 | |||
| e4c21c4d5c | |||
| d9e6dbf889 | |||
| 5f4469e306 | |||
| 16f6365b18 | |||
| cdab54bf51 | |||
| b6fea0c856 | |||
| 6f186932cc | |||
| de544ec725 | |||
| a6c0d2c6f1 | |||
| 437d956cad | |||
| 43ba0b9da6 | |||
| fe1e2c3b62 | |||
| bbe7e5f96d | |||
| 2bfad9fc65 | |||
| 9f24c18f7f | |||
| a2d1e5ea03 | |||
| b082e4e988 | |||
| 45fbce0c2a | |||
| ff7006e232 | |||
| c68fdef0e4 | |||
| 4c60e4edc1 | |||
| f2e4e59f8d | |||
| 898205b5b1 | |||
| 8ec6862861 | |||
| c3ba6322d8 | |||
| e589828425 | |||
| c909aa161b | |||
| 5cee7d44f1 | |||
| 973b323a99 | |||
| d55954a363 | |||
| adca151e4f | |||
| 43513049be | |||
| a5ca41131b | |||
| 2ea3810980 | |||
| c9cb54e8b2 | |||
| 23dd3db50b | |||
| 179c46914d | |||
| ddb18e6c21 | |||
| d2e147ba51 | |||
| b63033f846 | |||
| 677e1101e6 | |||
| 3abc2ccfbb | |||
| 827d28f1cd | |||
| 59d12a1bbe | |||
| 8134172301 | |||
| 8d983aa561 | |||
| bb3085ef84 | |||
| 78b1650bdb | |||
| 5edc72d475 | |||
| 1df32c9dfa | |||
| bedb1ff7f9 | |||
| fbf12f16b5 | |||
| 22a5f34e1f | |||
| e1a8b08619 | |||
| 466d30fb74 | |||
| 7a1c6d2918 | |||
| 9fcc0dc8e7 | |||
| 286f244caf | |||
| 52ebeab90b | |||
| 6be7591332 | |||
| 73325aaefb | |||
| 0adcc6a8d1 | |||
| 93fb6acd1d | |||
| 77d52b323d | |||
| ca8e8bf6ef | |||
| 9acac86ed5 | |||
| ba5007219d | |||
| ec046fd599 | |||
| 5452ff1153 | |||
| d03b1e0608 | |||
| 53c88ad5af | |||
| 21585ca565 | |||
| 2fabee647a | |||
| cf4632045c | |||
| ec75afbc44 | |||
| c7b298c46f | |||
| 3406bb5a4a | |||
| e9427b4694 | |||
| 517a1f1835 | |||
| 6e735bbd1e | |||
| 5a148aa3b9 | |||
| b4c362da37 | |||
| 575ae164c8 | |||
| a9606bfb4e | |||
| b5ec04723d | |||
| 51cc08411f | |||
| d75034deae | |||
| 4ce23debb6 | |||
| 063706a7bf | |||
| eb41a3655f | |||
| a84476187f | |||
| 70b603d601 | |||
| c9709f2698 | |||
| be4f479afd | |||
| a251465dbc | |||
| 9b7051f2be | |||
| 8bfdef79c4 | |||
| f2c2bf81b3 | |||
| 8b5247b9bb | |||
| 075b1dc0eb | |||
| 42e1f0478d | |||
| d4653678b2 | |||
| a4ce752e58 | |||
| f6649398ef | |||
| 02859cc270 | |||
| 4ed30e082f | |||
| d3985dd129 | |||
| ac70821fea | |||
| 38b273a1c9 | |||
| eb5c88fbb2 | |||
| 1102952b47 | |||
| 5ad6cadcee | |||
| 5d236808d6 | |||
| ada9243e84 | |||
| ad4e1c1b5b | |||
| c5105c29b0 | |||
| f689b0f3b2 | |||
| 730f614024 | |||
| 2e4eb17a48 | |||
| 55d2a1f09b | |||
| e3a5bcb907 | |||
| d56567c9de | |||
| d7c381e05d | |||
| 1d23dd2426 | |||
| 86ce00adf9 | |||
| e1eef013a8 | |||
| d20046c866 | |||
| 2df452fe5b | |||
| c31bfd8b94 | |||
| f443675f4f | |||
| a44bd8849d | |||
| 274c887140 | |||
| 44973ebd00 | |||
| 88f74163ff | |||
| 6cd57dd426 | |||
| 481e866011 | |||
| a78450ba79 | |||
| 9fcdeca692 | |||
| 8e10c56304 | |||
| 591f600b11 | |||
| af03e55a73 | |||
| 1462cddd1e | |||
| aac569a925 | |||
| d19ac1fd15 | |||
| 410a23751b | |||
| 8190507e8c | |||
| 645f74f39d | |||
| acdf0912d4 | |||
| 32e4e91ab8 | |||
| b59ca329f3 | |||
| beb9099bdc | |||
| a013d95f0f | |||
| 9d5daf0015 | |||
| 1146307736 | |||
| c25eaadc1d | |||
| 50f6e76ab9 | |||
| c3637e731f | |||
| c31eef6b82 | |||
| 802683b765 | |||
| 335cf93970 | |||
| 041954c067 | |||
| 2da44c3699 | |||
| 65e53092e8 | |||
| 0203aa2b6e | |||
| f83fe28a18 | |||
| e487b45898 | |||
| 4a94eab393 | |||
| 5ff7e6ef0e | |||
| 0c99f41bd9 | |||
| bcac810f71 | |||
| feae105426 | |||
| d46b9c54b1 | |||
| d0b7162b6a | |||
| c16660254b | |||
| bbe0d52740 | |||
| 65117ebdd7 | |||
| 445d55e800 | |||
| dbce751464 | |||
| b8640d903f | |||
| 6e7560ee77 | |||
| efa26a067f | |||
| f7cf7c198d | |||
| d32f4fc38e | |||
| 0c8b8647f3 | |||
| c38dbbb1d7 | |||
| 98cec15625 | |||
| bad9828f47 | |||
| 18f91ddffa | |||
| 335745d365 | |||
| 4204b31398 | |||
| 029a568645 | |||
| 6b2f1fcd3e | |||
| 3bdc610249 | |||
| c03a70fde2 | |||
| c77645e173 | |||
| 6531002d61 | |||
| fea808ca5f | |||
| 59ba408070 | |||
| 3a8931feef | |||
| 7ebd8f6bf5 | |||
| 73883979c6 | |||
| d8935b46b3 | |||
| 1505d04622 | |||
| 3b690cc31f | |||
| b3814920bd | |||
| 8bf1f828b9 | |||
| 911e69e3bc | |||
| 77b4a1eaf6 | |||
| 2ed12c429e | |||
| e578c52fdf | |||
| 5ff4e3c4ea | |||
| 1c2e7256c1 | |||
| 7a51ca225a | |||
| 8d242d8072 | |||
| 543b068efa | |||
| eadbd5e821 | |||
| c771f5a13c | |||
| f13b3111c3 | |||
| bb2714ff24 | |||
| 54c42b1fc2 | |||
| 1f42f933f0 | |||
| 6c533d225b | |||
| 67a89d1289 | |||
| 0b9bef2f38 | |||
| 1c4649409d | |||
| e1daaf07ce | |||
| cd21f2d1d7 | |||
| 836e41064f | |||
| 4658e4c739 | |||
| 7993a7cdb0 | |||
| 567cb7d737 | |||
| 985128b537 | |||
| cd35568e04 | |||
| f612509cac | |||
| f415190483 | |||
| e00733a346 | |||
| aafafa0e73 | |||
| c87c9af12e | |||
| 622215715f | |||
| c87250c028 | |||
| d6b6d700a5 | |||
| 9d4e2c98a3 | |||
| 08094c2660 | |||
| fda82c82b0 | |||
| f0eabd4ea0 | |||
| 5a4d812146 | |||
| bf156a13bd | |||
| 53d276a8fc | |||
| 978fa54518 | |||
| 31f82e58b5 | |||
| 5967f66e6d | |||
| bea81b54ca | |||
| 6fd403bdca | |||
| 3d673d9d40 | |||
| cac949de56 | |||
| dffa152698 | |||
| 7796298fca | |||
| 5291bfe8d4 | |||
| b364313297 | |||
| 2bef608e07 | |||
| aee13ad909 | |||
| 9d82eba599 | |||
| 4852beb390 | |||
| 522c2f61c0 | |||
| d331396afe | |||
| c725cee044 | |||
| 367ef4ecb2 | |||
| c3a64facd5 | |||
| 2671781e1b | |||
| 9291fa68aa | |||
| 6ebb3659f4 | |||
| 109696e965 | |||
| b86bbd370c | |||
| 1575a4fb1a | |||
| e2f500be90 | |||
| 284b00a826 | |||
| 66180e19b5 | |||
| 1531462d22 | |||
| e17cd1f298 | |||
| 13092e9f80 | |||
| 8133b8b9dd | |||
| e25aafac6d | |||
| 037c7beb1b | |||
| faac4dfc30 | |||
| 469a088a4d | |||
| 9c854f727f | |||
| 8f6e5bd24b | |||
| 992f91cf4c | |||
| 0c61d4c978 | |||
| 72d32edf9a | |||
| dde39def9e | |||
| 6b43007c44 | |||
| 876558cf77 | |||
| b35a146edf | |||
| 86cf6a9908 | |||
| b0f7288ac0 | |||
| 32fcc1a8fb | |||
| eb4d125eaf | |||
| 87e5cced3c | |||
| bcd9ee2d48 | |||
| edf3d87458 | |||
| 0c0c353ecc | |||
| aaa4c8f899 | |||
| 57e3565c11 | |||
| fbcf72d762 | |||
| ca8daa836e | |||
| c2ccdbec9d | |||
| 69aee36e75 | |||
| 063f5c3b55 | |||
| 6d1b8ca65e | |||
| 03899d4d9c | |||
| 2b84af977d | |||
| e15b180322 | |||
| f070030f6b | |||
| 330ac66b38 | |||
| 12a9e650af | |||
| 1e5ccd811e | |||
| 2902ee6ad5 | |||
| 90ce4fec2c | |||
| a7ab26d08d | |||
| dcc396afb7 | |||
| 3f1722d54d | |||
| c79658afbb | |||
| 6f84ebb323 | |||
| 54c8d62243 | |||
| 83e6476408 | |||
| 03f317ffdb | |||
| 3f67c7c74a | |||
| b8b4660563 | |||
| 3d42bfd479 | |||
| c4ebbaba74 | |||
| 2ae193092d | |||
| 7e1d52ff00 | |||
| c98f43b984 | |||
| e93f128a7a | |||
| 71d8e7edd2 | |||
| 48f4298a8d | |||
| 1c15beadc7 | |||
| 2c1600ddfb | |||
| 298f7d9d52 | |||
| 105f0bfde2 | |||
| cf3a78e114 | |||
| 9cc5f0f889 | |||
| 81e588a896 | |||
| e30db9ee77 | |||
| 235be757f8 | |||
| e31d26a887 | |||
| 2293ba02ea | |||
| 7188997dd1 | |||
| 31cfb09468 | |||
| b76f2e2008 | |||
| 4b90972341 | |||
| f4ff34224c | |||
| 877c9c4ff9 | |||
| ac0b7291dd | |||
| 491ef6085a | |||
| 3cedef4974 | |||
| 22ab04bd2b | |||
| e5a080aebe | |||
| c560cc5add | |||
| 0d27bc323b | |||
| c71d3cef18 | |||
| 4e2d8daa3a | |||
| d0f51da0af | |||
| aeb73bca27 | |||
| f239b03291 | |||
| 297c2965f4 | |||
| daddf4d98e | |||
| e05f9bfebf | |||
| ef46aeae6f | |||
| 7edb3fd856 | |||
| 43b79778ea | |||
| 37f1f53b56 | |||
| 67bd1cdcd9 | |||
| 506385e5a2 | |||
| 2d4586b1c4 | |||
| 1476b9cb9c | |||
| 7bdde68ece | |||
| 50f92f55e2 | |||
| 370db62bf0 | |||
| 65f34f1d31 | |||
| 00a3908abb | |||
| 32034d590a | |||
| 3635fb3910 | |||
| d2ecfe5491 | |||
| 1f759dce5b | |||
| ae41c6038b | |||
| f41f7eb2ad | |||
| d04f383161 | |||
| cb989d7489 | |||
| b5cba19d26 | |||
| b7271d7a46 | |||
| 768bdc2cc2 | |||
| a0a093e260 | |||
| 0b2a7fdc15 | |||
| f1876e20f8 | |||
| 7d6a6e53f7 | |||
| 6b765a1f77 | |||
| 3b3c93dd53 | |||
| 521083a309 | |||
| 6d35325601 | |||
| 3c65f37d84 | |||
| d75dd058d6 | |||
| 40475e02ec | |||
| f6ea9c1300 | |||
| 902359f24e | |||
| bb4d5f1e93 | |||
| 1dec3f000e | |||
| 6d89814795 | |||
| f339bc9f7f | |||
| bb80bc0c07 | |||
| 96677ff8bf | |||
| c7b6a6df79 | |||
| 8bb7e8bfb2 | |||
| 02ab343e22 | |||
| 4d875a18de | |||
| cff2336923 | |||
| 0e96bfdfa3 | |||
| a24ef48ad1 | |||
| fe9c4f3391 | |||
| 6cbb0739f8 | |||
| 79ebabfcfb | |||
| 0c8e3262fe | |||
| c24a040c19 | |||
| 4f39cb8dfa | |||
| cdd2816642 | |||
| 27b6dfa4d2 | |||
| 204cbd0209 | |||
| b7980aad5a | |||
| e175729e2c | |||
| c26ad4c807 | |||
| 4372adc703 | |||
| 8a0c2b9b13 | |||
| 4443a1c030 | |||
| 39a02235cf | |||
| db89561480 | |||
| a4cbb11693 | |||
| 1ceeacc526 | |||
| b59052cc43 | |||
| b5d8161bc2 | |||
| b497eda26e | |||
| 44019e1042 | |||
| fd0e1da4a2 | |||
| f6c67b475a | |||
| ea18a5ad15 | |||
| 4d0cd3f497 | |||
| 7dbdeaebe0 | |||
| 2085bcceb6 | |||
| c09c962cb6 | |||
| 9108459ae4 | |||
| 992bac0b1f | |||
| ebd6917a1d | |||
| 3e079e3b80 | |||
| 2ca20be197 | |||
| 17f23f3751 | |||
| 8e3d699856 | |||
| f1a168fa53 | |||
| 3575113655 | |||
| fe9dd7d23f | |||
| 9feb9d04b3 | |||
| 5419b1439a | |||
| 6f8fbe3f09 | |||
| e4489343fe | |||
| d9f4a5793d | |||
| 70fcdc9ebb | |||
| 78e7a81638 | |||
| 58e82d5dbd | |||
| 06d15be43a | |||
| e1e7011853 | |||
| eff7645035 | |||
| eb75e52278 | |||
| 15e6148272 | |||
| ccd448a675 | |||
| db54c019ad | |||
| b762b4d72c | |||
| 2f8faa839d | |||
| 831c325c63 | |||
| f4f73078c5 | |||
| f7d43ad5af | |||
| a77c777980 | |||
| a34db7449e | |||
| 0283bd2f97 | |||
| a8de2f8ae7 | |||
| d5dee75df3 | |||
| 6b7631ed5e | |||
| 79cb5c0631 | |||
| 7d9901540f | |||
| e979e9c9fb | |||
| de719df6fe | |||
| 38d7f91ea0 | |||
| a20a429e8c | |||
| 9b63fb4ee2 | |||
| 099efdbc1d | |||
| af9120fc7a | |||
| 798a48aa96 | |||
| 462e22a3b0 | |||
| 4e432ed03f | |||
| 1b56c0f191 | |||
| 94cbeba495 | |||
| dfa74a69f7 | |||
| 9e1e4eeec2 | |||
| 962f8233b0 | |||
| 31923d511e | |||
| fdbb8300d3 | |||
| 203d8bca57 | |||
| 74c331eaf7 | |||
| 54365528a8 | |||
| bc174f7054 | |||
| 221e068bac | |||
| 1bdceeecf4 | |||
| a6824d9cd0 | |||
| fe03f9942b | |||
| 4c196922fb | |||
| 2a9a513d85 | |||
| 2bcea27ecd | |||
| fb7341f1f7 | |||
| f327daa12d | |||
| 2872b9fbf9 | |||
| cedd5c9c96 | |||
| 60e6aa9b54 | |||
| 541f482518 | |||
| 4019b7939a | |||
| 013b9c4c7c | |||
| 79addfda42 | |||
| 8546bda471 | |||
| 0770f174a1 | |||
| 5f4a89cecc | |||
| cbe0b1c5a6 | |||
| 0af193c505 | |||
| fdcfcc77a0 | |||
| 06d166d0d7 | |||
| b1b3e39fcd | |||
| 5ec025a3b9 | |||
| 58b7fbcf75 | |||
| be053d47e4 | |||
| fae1981161 | |||
| fd95549de9 | |||
| ff10bc05ec | |||
| eb8cd53de2 | |||
| 3fc863561a | |||
| 131cd94495 | |||
| f3a90a63b6 | |||
| 2494173aec | |||
| 866eb6241b | |||
| 86b3df1941 | |||
| e87f6d56f5 | |||
| acc890730f | |||
| b0707739fd | |||
| 251dd1fe45 | |||
| b9f3dc65e0 | |||
| 238ad7ce51 | |||
| 99fd5fca4d | |||
| 8eda77b76d | |||
| 81ac240ac8 | |||
| 6109798fab | |||
| 95715a007d | |||
| b33ec201ac | |||
| b53fbaf5b3 | |||
| 1e03a2e553 | |||
| fda7c6f67a | |||
| fabb7982ff | |||
| cbf206be60 | |||
| aa0c282205 | |||
| 9759365329 | |||
| e3738f6422 | |||
| 9746d169f9 | |||
| 2e6d03ff00 | |||
| f7b7d3d65e | |||
| 4037cf11aa | |||
| 02aeb321ce | |||
| 0012619257 | |||
| 6f3ade0d94 | |||
| cf572f328a | |||
| d1ce36038c | |||
| b382351c7b | |||
| 4e5e862f58 | |||
| ab84835362 | |||
| 41ce8489dc | |||
| ef3faf5832 | |||
| edf089ec9e | |||
| 0ae9a3605c | |||
| 7f9c4e52ac | |||
| 35947f96a8 | |||
| b0f91f1eea | |||
| 13dfca1749 | |||
| 9d9cd8a362 | |||
| 5e5c41fda5 | |||
| 3ebdc52b3e | |||
| af54f48cec | |||
| 8656059151 | |||
| a6d38f2458 | |||
| 085bdf5cfa | |||
| 6883bcacee | |||
| 2ecc6e0368 | |||
| 8fb5ca2fe1 | |||
| e40345095f | |||
| ffc0c7bb7b | |||
| 58fadc8928 | |||
| d96a607c04 | |||
| 2ea2c8c05f | |||
| b15f514018 | |||
| 05a33a0ec9 | |||
| 747d266742 | |||
| 522d30545b | |||
| 6135a44a8d | |||
| 7c7d646792 | |||
| 4a36fd2ec3 | |||
| b1bcc287cb | |||
| 6f5868a9d7 | |||
| 75863441f4 | |||
| 9763cb00e5 | |||
| 521599ef39 | |||
| 1921a64f4b | |||
| 6b73f5d555 | |||
| e0408f30ba | |||
| dca44fa093 | |||
| bbacb76581 | |||
| 1da8617a53 | |||
| e5967f7e9d | |||
| 65d84f9e9d | |||
| 93e9498b41 | |||
| 95332d5db9 | |||
| 9864792bbf | |||
| ca9d1eed7a | |||
| 38e867c917 | |||
| 1ff6daaa27 | |||
| 3ee1dbb8a5 | |||
| b4571d5c98 | |||
| 29d49d72f9 | |||
| 81de0fc7e4 | |||
| 9d5d266d2a | |||
| b97935299f | |||
| 32a7ea1c16 | |||
| 9fd95e6a1e | |||
| 61ba83c775 | |||
| 6369fed5fc | |||
| 42f4d1477d | |||
| 609ac9c9a2 | |||
| 79f2367472 | |||
| dfc9362084 | |||
| 487b469603 | |||
| 19e1df1e5d | |||
| fc55010888 | |||
| 56a36aa595 | |||
| 0b3158fdd5 | |||
| 896cd950e9 | |||
| af5e1b805f | |||
| 3f9943270c | |||
| 902d246d1a | |||
| 785bee2b39 | |||
| 4b335db31c | |||
| 4bef527ebb | |||
| 8273031d7e | |||
| dbf69bcd98 | |||
| 24d3096752 | |||
| 9a3754fbf8 | |||
| c2a95a13fe | |||
| b46466ac96 | |||
| 06991ddb17 | |||
| 4b3f4a868a | |||
| 014eff3534 | |||
| c01b7ddb59 | |||
| 0ff700849f | |||
| 5c695dea20 | |||
| c7ee4ca4db | |||
| c3da026b33 | |||
| 98da4e1791 | |||
| 8626b6d9f2 | |||
| 80c5331a5d | |||
| 39dc5c8160 | |||
| f3002e4fb6 | |||
| c451823c2b | |||
| b37cffd704 | |||
| 2af91dbf2a | |||
| f2551318fc | |||
| 22eb84f944 | |||
| 1ece0915f1 | |||
| 87853a2015 | |||
| 46a1b74799 | |||
| 0f6e7e5eab | |||
| 5dfa9615d2 | |||
| 1bde777bee | |||
| fa4f5df3e7 | |||
| 8a3c3810e0 | |||
| 144532530a | |||
| 0f1129e19b | |||
| 1f74580f15 | |||
| f93ba9970c | |||
| f87a3d0892 | |||
| c661ad67d0 | |||
| ce4dc9e3fa | |||
| 3d2c6e6032 | |||
| 6000a0cfe3 | |||
| b80c60997a | |||
| 35e45f0df1 | |||
| e65f5b9f78 | |||
| 5969f71e67 | |||
| b1307863eb | |||
| 9d0abe993b | |||
| c53bb7cf67 | |||
| 0cea26c628 | |||
| 610c919c72 | |||
| 2c35f94f7c | |||
| cd9a3870b3 | |||
| e11373f23a | |||
| f591635fc1 | |||
| 474b57ca61 | |||
| 8671887abc | |||
| 8274d1baa5 | |||
| bde601bfff | |||
| a2e0951042 | |||
| 3c9a8a38dd | |||
| 4c067fd39f | |||
| 5c251ee774 | |||
| ddda691552 | |||
| ba73090d53 | |||
| a080b606ab | |||
| 7c0f43c8a3 | |||
| 4fad1aee6b | |||
| 19aec5bc8d | |||
| 33ee60736c | |||
| c1bccb970f | |||
| 481cc029fa | |||
| bdaf58a3c4 | |||
| 0f64671dc0 | |||
| 60f055f293 | |||
| c67a9215e3 | |||
| a0e9df6d6d | |||
| 8341749c04 | |||
| 66d1886663 | |||
| 710e1fc278 | |||
| 4cf98584da | |||
| 3fb3cee423 | |||
| 2d1504a057 | |||
| 4fcfd089d8 | |||
| 04422a4637 | |||
| 37e6548246 | |||
| a761989f3e | |||
| acaa8b1731 | |||
| 082f47663d | |||
| 92f42154d5 | |||
| fc1084ce33 | |||
| adc3ab7e0a | |||
| dcc8c56969 | |||
| 0b3472d227 | |||
| b50121ad0b | |||
| dfddfc3e06 | |||
| 34ec6210c6 | |||
| daaef316e9 | |||
| cdac12bb2f | |||
| 3ab99647aa | |||
| 529482a83e | |||
| 29906ec057 | |||
| 9296ba7492 | |||
| 821c6d807d | |||
| 991b741cbe | |||
| 2559f0e822 | |||
| 8bb1ed3e95 | |||
| 56ba3fcb92 | |||
| e99a20a120 | |||
| f1a25b21a6 | |||
| cf9595ce58 | |||
| 7feece597a | |||
| fa16c782ca | |||
| a03d0b6a4a | |||
| dff76b8912 | |||
| cffea9a9bc | |||
| 43fee42198 | |||
| 5cd3968929 | |||
| 65dcae79f8 | |||
| 2b3b75a4a5 | |||
| 26b395110c | |||
| e11b3becfd | |||
| 73fa937f5c | |||
| 6ebe2e54ac | |||
| fb29a11cc9 | |||
| a9e06cbf92 | |||
| 93017c044d | |||
| c223ddbb9a | |||
| f00aeacb8b | |||
| 5b49071d6b | |||
| 17053a882b | |||
| 5e723d31a4 | |||
| 3283bd8b75 | |||
| 770d3c0015 | |||
| d15dfafd5d | |||
| 545c13d55c | |||
| e2099ac9ca | |||
| c937583a50 | |||
| 43c7a19849 | |||
| 83543487e7 | |||
| 434b259525 | |||
| add8efaba8 | |||
| 12ed79ca60 | |||
| 1e863382d3 | |||
| bad3504d4a | |||
| d94f207162 | |||
| 2c4b7781a4 | |||
| 4574c6ff07 | |||
| 7b5043e87b | |||
| a06f3ac5da | |||
| 721346a40a | |||
| f252871fb8 | |||
| 107196122c | |||
| 563c02d8da | |||
| 765934970a | |||
| 9cbdfda829 | |||
| c1fbc8cd68 | |||
| a92107cc47 | |||
| 3e84e116e8 | |||
| 7c0cdd169e | |||
| 424fd96615 | |||
| ebfcea88da | |||
| 3c7eb2f5e2 | |||
| 936167972f | |||
| 7f6070c960 | |||
| 0aea9c129c | |||
| d20fb7daa8 | |||
| a619f8a2fe | |||
| 0acb858d7b | |||
| e459be76fe | |||
| c4c59ccc75 | |||
| c820315409 | |||
| 2a19b61b7a | |||
| e1cf64ae16 | |||
| d3c2f8eb43 | |||
| a00453c83a | |||
| 2eb0e54909 | |||
| ac87bc57e9 | |||
| 2b8ea857f0 | |||
| 11c52114b2 | |||
| f55f9b4dd3 | |||
| cdd369ea98 | |||
| f2aab9f476 | |||
| 2619dc3556 | |||
| 1bbed351ba | |||
| 4cfb2644c6 | |||
| 5b85c7ad39 | |||
| ab3a3156f2 | |||
| 28a582025e | |||
| 8451a83a3a | |||
| 92c8dcc666 | |||
| da68b02e1e | |||
| 2d7729dbe9 | |||
| 6467edb843 | |||
| 1f01b3a9ff | |||
| 8aa1f8926d | |||
| ace363fa35 | |||
| 919f70a5fd | |||
| 9d6ad771a3 | |||
| bafccb20c6 | |||
| cef30c2af0 | |||
| ca58056a75 | |||
| 03e2e99498 | |||
| fba7afc4e9 | |||
| d7dda336ec | |||
| 55d4395160 | |||
| f7d5baa6d0 | |||
| 6ff509d263 | |||
| 57778981a7 | |||
| 6ac3bc564f | |||
| 82d08e2153 | |||
| 6212cd77e8 | |||
| 170034f943 | |||
| e639a8f9f1 | |||
| 9d10c45dac | |||
| b84159f2f1 | |||
| 49f26b4049 | |||
| 0e7e44cee2 | |||
| 36e769502c | |||
| a4b6580247 | |||
| 84fb1c5127 | |||
| ddfd0fb81d | |||
| 37edbf5824 | |||
| e15212bf49 | |||
| 6a0cc1b1f3 | |||
| 0e8339c701 | |||
| 5d71a4dbde | |||
| 0b78030c59 | |||
| 24237c16bf | |||
| c48da5dea7 | |||
| 6702ca10a1 | |||
| 4b44bd5e61 | |||
| 8a55beda92 | |||
| a12b824339 | |||
| c4a743189e | |||
| 85f9ef35f6 | |||
| 6de220e38a | |||
| 0d455d8c2f | |||
| f7b0b44ef6 | |||
| 81282a9c88 | |||
| a9b302e38d | |||
| 1fe4c367f7 | |||
| 2de7583900 | |||
| 356703c83e | |||
| 1cae709b2b | |||
| 46a492248f | |||
| d876ea6711 | |||
| b40b4c3cfd | |||
| 44980d6c46 | |||
| 442f9647a2 | |||
| a06ef07178 | |||
| 0c2ea5da4c | |||
| 45814ceb49 | |||
| 41f4617e66 | |||
| a463711b03 | |||
| 3a147141b1 | |||
| aea1c13bd3 | |||
| 9cc4c017ae | |||
| 88022747be | |||
| ebb292a2f7 | |||
| 818998259d | |||
| 36b02c2cec | |||
| e6195ade3e | |||
| 231a875bb4 | |||
| 378c777a38 | |||
| 8ef63916ef | |||
| f32ecdf5f1 | |||
| 94739b9b8e | |||
| 023db4e04e | |||
| 5a4b95f5fe | |||
| b091657b5c | |||
| f7bf5c9328 | |||
| 86e521b9aa | |||
| e08cf57b72 | |||
| 9e06cb9a83 | |||
| c65e8622b8 | |||
| 7795efeb7a | |||
| e725e0020e | |||
| 8478ce25f1 | |||
| 22cdac6210 | |||
| 3422a1a59f | |||
| f807b8cb46 | |||
| e1e510ce1e | |||
| 36bc3ff22d | |||
| 1db1ffde99 | |||
| 7984b625ba | |||
| bb22f062ed | |||
| a3086e6a5b | |||
| 1eb9bd34fd | |||
| cff7baaaad | |||
| 47af700375 | |||
| eb7f53a1e3 | |||
| d23792fda2 | |||
| b5cbb8e450 | |||
| fc037b4518 | |||
| c04921f42b | |||
| 8af3463668 | |||
| 094565ccd6 | |||
| 07b9769504 | |||
| 566b12f5d1 | |||
| a560999d13 | |||
| a818a3d293 | |||
| 4d68a174cb | |||
| 905219e523 | |||
| c675b87040 | |||
| e2dadfdc40 | |||
| a66f9aa04d | |||
| 863e74dd2e | |||
| aebb07c5cc | |||
| 677fec0a0b | |||
| 61f06faaf5 | |||
| fcf8309c23 | |||
| 76b19a4980 | |||
| ed1a9fc7aa | |||
| b71e30755d | |||
| fe196d1b20 | |||
| 0b152a3cb8 | |||
| 1a0e096ddb | |||
| bf040d4c42 | |||
| 3e2101aa5b | |||
| 44f11b38e7 | |||
| 06f8514bc1 | |||
| d9a9f1c25c | |||
| e77f7244ba | |||
| 09779cd1e1 | |||
| 11024168db | |||
| 304914513e | |||
| 03d0efcfc6 | |||
| 0c2bdc9146 | |||
| 188450b0c0 | |||
| ddf6bbfa46 | |||
| 2c7c98a152 | |||
| d31ac75718 | |||
| 4b28c659de | |||
| 00b937e52a | |||
| 68f333fb87 | |||
| 085b4d9319 | |||
| b8edd14f39 | |||
| 61a19d694b | |||
| aa96859798 | |||
| abf015f485 | |||
| 0b9a02afde | |||
| e332ce28f8 | |||
| 08e779f9f1 | |||
| a53b6cd28f | |||
| 47ebab237b | |||
| 5a5af60f97 | |||
| 50cc17c7cb | |||
| a1e504c138 | |||
| 4cc413047c | |||
| 2397097e4d | |||
| c88f959ec9 | |||
| 0b2e28b62d | |||
| c7f2ead696 | |||
| b454e02d01 | |||
| 47df2ffc3e | |||
| d18e431e2f | |||
| 0a147d2db7 | |||
| ccdc933064 | |||
| 023f2d4569 | |||
| 06a7371d2b | |||
| 626f5d3487 | |||
| 8cd3b9fe2e | |||
| 716c35d52a | |||
| 8cc0f3918b | |||
| 98b51f0799 | |||
| 81d6dad548 | |||
| 41bc11cf96 | |||
| 721dbe415a | |||
| f5c0b51428 | |||
| 892c6ad80c | |||
| a47805e494 | |||
| 9dd49054d1 | |||
| f5d1d1a0b7 | |||
| d75fcb7fec | |||
| 826be45b6a | |||
| d35d9c17c5 | |||
| f9553e7d44 | |||
| 638c9720cf | |||
| 08a190882f | |||
| bfa7530a39 | |||
| 6c74148c27 | |||
| 480cad0fed | |||
| 32be489136 | |||
| 11801d8e2e | |||
| 1b280a2940 | |||
| 424890a1e1 | |||
| 5f85219495 | |||
| a63d687f1c | |||
| f2d6c3ad83 | |||
| 0b6941d5ce | |||
| 048696ee93 | |||
| ae51676471 | |||
| 27a405fb1d | |||
| e2cf65b591 | |||
| 9749fc817d | |||
| e6600f2c43 | |||
| a664931e7a | |||
| a2ba965600 | |||
| 65255dbb50 | |||
| a5cb8761a5 | |||
| e3930e0717 | |||
| afd59e9933 | |||
| 8087524bef | |||
| 605440812f | |||
| b10c6eb615 | |||
| a96264ff6a | |||
| deb3893820 | |||
| 9b1d822b5b | |||
| f933fb705c | |||
| 5cd61c4c02 | |||
| 586fa70eac | |||
| 9b420ad33f | |||
| 5891290672 | |||
| 72a7b51d47 | |||
| 2943e0e58d | |||
| 5abce916a8 | |||
| 89d4be8a0a | |||
| b7113bda23 | |||
| 0088929622 | |||
| b3468cf7f2 | |||
| f88c5c8528 | |||
| 687fdda7f7 | |||
| aec51e514c | |||
| 308d4600ef | |||
| 50a5fa15bb | |||
| 7d96a57d73 | |||
| 162ebfd4e0 | |||
| a586a92d5e | |||
| 3df20a924f | |||
| ddcf466e4e | |||
| 5d10cbf18d | |||
| 918ea59b9a | |||
| 8d9afa7592 | |||
| 95e05336c2 | |||
| a188385817 | |||
| 0a6baf331b | |||
| 0e29e052d5 | |||
| d8d255980e | |||
| dc5a5fa543 | |||
| 8638fc91ff | |||
| 96a0900edc | |||
| abea80e3ab | |||
| 42dfe936b7 | |||
| 8385bcc2d7 | |||
| 9b8f60b64b | |||
| b9dab77c8b | |||
| 474114236e | |||
| 238b0b421a | |||
| 8abe62886a | |||
| 78cc9cffe4 | |||
| 59a5dd713f | |||
| a39024ff03 | |||
| 72bfbd93a8 | |||
| c9a3e3d9d2 | |||
| 8387708901 | |||
| b565b4b3b9 | |||
| 893dcd4f24 | |||
| d613aa8f3e | |||
| 5750bb7067 | |||
| 0e07ae6ce8 | |||
| 02b6351e13 | |||
| 78367af830 | |||
| dc05cd481f | |||
| 7daad5477a | |||
| 45cdfbfae8 | |||
| 4159534a64 | |||
| 3fb5c38571 | |||
| 59f80ebc47 | |||
| 198a97b00c | |||
| 3ea4e917e8 | |||
| 60ad077172 | |||
| 356ad28e41 | |||
| e241141220 | |||
| 14bb1b467a | |||
| 2bbea6fd3f | |||
| 48aef25b3f | |||
| 8e50e5dee3 | |||
| d5d54d4d3b | |||
| 412e8a32dd | |||
| 0f82cf409b | |||
| 79df39acab | |||
| d00177a9b6 | |||
| 8786bae7dc | |||
| 4b3f8ca361 | |||
| 03183218f7 | |||
| 95b6db57e1 | |||
| bbe0c2457b | |||
| c894c53e69 | |||
| 5b3fb7387d | |||
| feac310caf | |||
| d67ec3feb3 | |||
| cf8abb4528 | |||
| d66de26de4 | |||
| e5a7ada3cf | |||
| b76fdd7fe4 | |||
| 7edf3f6147 | |||
| 2143dff2ae | |||
| 32c714d1b6 | |||
| 84e699ee24 | |||
| 7fdb572b8b | |||
| 91ffb0820a | |||
| cfd3b66be9 | |||
| 75c4f9dea8 | |||
| a76a32230d |
@@ -0,0 +1,11 @@
|
|||||||
|
#
|
||||||
|
# http://editorconfig.org
|
||||||
|
#
|
||||||
|
|
||||||
|
root = true
|
||||||
|
|
||||||
|
[*]
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 2
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
# These are supported funding model platforms
|
||||||
|
|
||||||
|
github: greper
|
||||||
|
buy_me_a_coffee: greper
|
||||||
|
custom: ['https://afdian.com/a/greper']
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
---
|
||||||
|
name: Plugin Apply
|
||||||
|
about: 部署插件申请支持
|
||||||
|
title: "[Plugin] "
|
||||||
|
labels: feature
|
||||||
|
---
|
||||||
|
|
||||||
|
> > 感谢您支持certd,请按如下规范提交issue
|
||||||
|
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
||||||
|
|
||||||
|
# 新部署插件申请支持
|
||||||
|
|
||||||
|
## 1. 需求描述
|
||||||
|
`请在此处简要描述你的需求`
|
||||||
|
|
||||||
|
|
||||||
|
## 2. 要部署证书应用的信息
|
||||||
|
|
||||||
|
1. 应用名称:
|
||||||
|
|
||||||
|
|
||||||
|
2. 应用网址/项目地址/官方网站:
|
||||||
|
|
||||||
|
|
||||||
|
3. 管理证书界面截图(或者手动部署证书方式介绍及截图):
|
||||||
|
|
||||||
|
|
||||||
|
4. 是否有API接口,接口地址:
|
||||||
|
|
||||||
|
|
||||||
|
5. 如果没有API接口,网页登录是否需要验证码:
|
||||||
|
|
||||||
|
|
||||||
|
6. 是否可以提供测试账号?(如果可以请留下联系方式或者加作者好友)
|
||||||
|
|
||||||
|
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
---
|
||||||
|
name: DNS Provider Apply
|
||||||
|
about: 域名提供商申请支持
|
||||||
|
title: "[DNS] "
|
||||||
|
labels: feature
|
||||||
|
---
|
||||||
|
|
||||||
|
|
||||||
|
> 感谢您支持certd,请按如下规范提交issue
|
||||||
|
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
||||||
|
|
||||||
|
# 新域名提供商支持申请
|
||||||
|
|
||||||
|
## 1. 基本信息
|
||||||
|
请填写如下内容:
|
||||||
|
|
||||||
|
1. 域名提供商名称:
|
||||||
|
|
||||||
|
|
||||||
|
2. 管理页面地址:
|
||||||
|
|
||||||
|
|
||||||
|
3. 是否有API接口,接口地址:
|
||||||
|
|
||||||
|
|
||||||
|
4. 如果没有API接口,网页登录是否有验证码:
|
||||||
|
|
||||||
|
|
||||||
|
5. 是否可以提供测试账号?(如果可以请留下联系方式或者加作者好友)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## 2. 截图
|
||||||
|
|
||||||
|
`域名管理页面截图`
|
||||||
|
|
||||||
@@ -1,21 +1,28 @@
|
|||||||
> 感谢您支持certd,请按如下规范提交issue
|
---
|
||||||
|
name: Bug Report
|
||||||
|
about: 错误或问题报告
|
||||||
|
title: "[BUG] "
|
||||||
|
labels: bug
|
||||||
|
---
|
||||||
|
|
||||||
|
|
||||||
|
> 感谢您支持certd,请按如下规范提交issue
|
||||||
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
||||||
|
|
||||||
|
# bug提交
|
||||||
## 一、问题描述
|
## 1、问题描述
|
||||||
`请在此处简要描述你所遇到的问题,必要时请贴出相关截图辅助理解和定位`
|
`请在此处简要描述你所遇到的问题,必要时请贴出相关截图辅助理解和定位`
|
||||||
|
|
||||||
### 复现步骤
|
### 2、复现步骤
|
||||||
`请描述复现问题的详细步骤`
|
`请描述复现问题的详细步骤`
|
||||||
`如果非示例页面的问题,最好能提供最小复现示例的代码、或者仓库链接`
|
`如果非示例页面的问题,最好能提供最小复现示例的代码、或者仓库链接`
|
||||||
|
|
||||||
|
|
||||||
### 报错截图
|
### 3.报错截图
|
||||||
`请贴出报错日志截图`
|
`请贴出报错日志截图`
|
||||||
|
|
||||||
### 效果截图
|
### 4、效果截图
|
||||||
`请贴出效果截图`
|
`请贴出效果截图`
|
||||||
#### 1. 期望效果
|
#### 4.1. 期望效果
|
||||||
|
|
||||||
#### 2. 实际效果
|
|
||||||
|
|
||||||
|
#### 4.2. 实际效果
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
---
|
||||||
|
name: Feature Request
|
||||||
|
about: 新需求、新特性申请支持
|
||||||
|
title: "[Feature] "
|
||||||
|
labels: feature
|
||||||
|
---
|
||||||
|
|
||||||
|
> > 感谢您支持certd,请按如下规范提交issue
|
||||||
|
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
||||||
|
|
||||||
|
|
||||||
|
# 新特性申请
|
||||||
|
>注意:这里仅供如果是要申请新的部署插件,请提交插件申请
|
||||||
|
|
||||||
|
## 1. 需求描述,需求背景
|
||||||
|
`请在此处简要描述你所遇到的问题,必要时请贴出相关截图辅助理解`
|
||||||
|
|
||||||
|
|
||||||
|
## 2. 期望效果
|
||||||
|
`必要时可以截图描述你的期望效果`
|
||||||
|
|
||||||
|
|
||||||
|
## 3. 你的解决方案
|
||||||
|
`如果你有解决方案,请描述你的方案`
|
||||||
@@ -3,7 +3,7 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches: ['v2-dev']
|
branches: ['v2-dev']
|
||||||
paths:
|
paths:
|
||||||
- "build.trigger"
|
- "trigger/build.trigger"
|
||||||
|
|
||||||
# schedule:
|
# schedule:
|
||||||
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
||||||
@@ -68,7 +68,7 @@ jobs:
|
|||||||
registry: ghcr.io
|
registry: ghcr.io
|
||||||
username: ${{ github.actor }}
|
username: ${{ github.actor }}
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
- name: Login to Docker Hub
|
- name: Login to Docker Hub
|
||||||
uses: docker/login-action@v3
|
uses: docker/login-action@v3
|
||||||
with:
|
with:
|
||||||
@@ -82,34 +82,4 @@ jobs:
|
|||||||
push: true
|
push: true
|
||||||
context: ./packages/ui/
|
context: ./packages/ui/
|
||||||
tags: |
|
tags: |
|
||||||
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest
|
|
||||||
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}
|
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}
|
||||||
greper/certd:latest
|
|
||||||
greper/certd:${{steps.get_certd_version.outputs.result}}
|
|
||||||
ghcr.io/${{ github.repository }}:latest
|
|
||||||
ghcr.io/${{ github.repository }}:${{steps.get_certd_version.outputs.result}}
|
|
||||||
- name: Build armv7
|
|
||||||
uses: docker/build-push-action@v6
|
|
||||||
with:
|
|
||||||
platforms: linux/arm/v7
|
|
||||||
push: true
|
|
||||||
context: ./packages/ui/
|
|
||||||
tags: |
|
|
||||||
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:armv7
|
|
||||||
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}-armv7
|
|
||||||
greper/certd:armv7
|
|
||||||
greper/certd:${{steps.get_certd_version.outputs.result}}-armv7
|
|
||||||
ghcr.io/${{ github.repository }}:armv7
|
|
||||||
ghcr.io/${{ github.repository }}:${{steps.get_certd_version.outputs.result}}-armv7
|
|
||||||
|
|
||||||
# - name: Build agent
|
|
||||||
# uses: docker/build-push-action@v6
|
|
||||||
# with:
|
|
||||||
# platforms: linux/amd64,linux/arm64
|
|
||||||
# push: true
|
|
||||||
# context: ./packages/ui/agent/
|
|
||||||
# tags: |
|
|
||||||
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-agent:latest
|
|
||||||
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-agent:${{steps.get_certd_version.outputs.result}}
|
|
||||||
# greper/certd-agent:latest
|
|
||||||
# greper/certd-agent:${{steps.get_certd_version.outputs.result}}
|
|
||||||
|
|||||||
@@ -3,12 +3,13 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches: ['v2-dev']
|
branches: ['v2-dev']
|
||||||
paths:
|
paths:
|
||||||
- "deploy.trigger"
|
- "trigger/deploy.trigger"
|
||||||
workflow_run:
|
workflow_run:
|
||||||
workflows: [ "build-image" ]
|
workflows: [ "build-image" ]
|
||||||
types:
|
types:
|
||||||
- completed
|
- completed
|
||||||
|
|
||||||
|
|
||||||
# schedule:
|
# schedule:
|
||||||
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
||||||
# - cron: '17 19 * * *'
|
# - cron: '17 19 * * *'
|
||||||
@@ -18,6 +19,7 @@ permissions:
|
|||||||
jobs:
|
jobs:
|
||||||
deploy-certd-demo:
|
deploy-certd-demo:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
if: ${{ github.event.workflow_run.conclusion == 'success' }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout Code
|
- name: Checkout Code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
@@ -43,7 +45,8 @@ jobs:
|
|||||||
- name: deploy-certd-demo
|
- name: deploy-certd-demo
|
||||||
uses: tyrrrz/action-http-request@master
|
uses: tyrrrz/action-http-request@master
|
||||||
with:
|
with:
|
||||||
url: http://flow-openapi.aliyun.com/pipeline/webhook/lzCzlGrLCOHQaTMMt0mG
|
# 通过webhook 触发 certd-demo来部署
|
||||||
|
url: ${{ secrets.WEBHOOK_CERTD_DEMO }}
|
||||||
method: POST
|
method: POST
|
||||||
headers: |
|
headers: |
|
||||||
Content-Type: application/json
|
Content-Type: application/json
|
||||||
@@ -53,15 +56,3 @@ jobs:
|
|||||||
}
|
}
|
||||||
retry-count: 3
|
retry-count: 3
|
||||||
retry-delay: 5000
|
retry-delay: 5000
|
||||||
|
|
||||||
- name: deploy-certd-doc
|
|
||||||
uses: tyrrrz/action-http-request@master
|
|
||||||
with:
|
|
||||||
url: http://flow-openapi.aliyun.com/pipeline/webhook/IiSxLDp9aOhgDUxJPytv
|
|
||||||
method: POST
|
|
||||||
body: |
|
|
||||||
{}
|
|
||||||
headers: |
|
|
||||||
Content-Type: application/json
|
|
||||||
retry-count: 3
|
|
||||||
retry-delay: 5000
|
|
||||||
|
|||||||
@@ -0,0 +1,61 @@
|
|||||||
|
name: publish-atomgit
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: ['v2-dev']
|
||||||
|
paths:
|
||||||
|
- "trigger/publish.trigger"
|
||||||
|
workflow_run:
|
||||||
|
workflows: [ "build-image-for-release" ]
|
||||||
|
types:
|
||||||
|
- completed
|
||||||
|
|
||||||
|
# schedule:
|
||||||
|
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
||||||
|
# - cron: '17 19 * * *'
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
packages: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
publish-atomgit:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: ${{ github.event.workflow_run.conclusion == 'success' }}
|
||||||
|
steps:
|
||||||
|
- name: Checkout Code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
lfs: true
|
||||||
|
- name: get_certd_version
|
||||||
|
id: get_certd_version
|
||||||
|
uses: actions/github-script@v6
|
||||||
|
with:
|
||||||
|
result-encoding: string
|
||||||
|
script: |
|
||||||
|
const fs = require('fs');
|
||||||
|
const path = require('path');
|
||||||
|
const pnpmWorkspace = "./pnpm-workspace.yaml";
|
||||||
|
fs.unlinkSync(pnpmWorkspace)
|
||||||
|
const jsonFilePath = "./packages/ui/certd-server/package.json";
|
||||||
|
const jsonContent = fs.readFileSync(jsonFilePath, 'utf-8');
|
||||||
|
const pkg = JSON.parse(jsonContent)
|
||||||
|
console.log("certd_version:",pkg.version);
|
||||||
|
return pkg.version
|
||||||
|
- run: |
|
||||||
|
npm install -g pnpm
|
||||||
|
pnpm install
|
||||||
|
npm run build
|
||||||
|
working-directory: ./packages/ui/certd-client
|
||||||
|
|
||||||
|
- name: publish_to_atomgit
|
||||||
|
id: publish_to_atomgit
|
||||||
|
run: |
|
||||||
|
rootDir=$(pwd)
|
||||||
|
rm -rf ./packages/ui/certd-client/dist/**/*.gz
|
||||||
|
cd ./packages/ui/certd-client/dist && zip -r ../ui.zip .
|
||||||
|
cd $rootDir
|
||||||
|
export ATOMGIT_TOKEN=${{ secrets.ATOMGIT_TOKEN }}
|
||||||
|
pnpm install
|
||||||
|
npm run publish_to_atomgit
|
||||||
|
working-directory: ./
|
||||||
|
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
name: publish-gitee
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: ['v2-dev']
|
||||||
|
paths:
|
||||||
|
- "trigger/publish.trigger"
|
||||||
|
workflow_run:
|
||||||
|
workflows: [ "build-image-for-release" ]
|
||||||
|
types:
|
||||||
|
- completed
|
||||||
|
|
||||||
|
# schedule:
|
||||||
|
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
||||||
|
# - cron: '17 19 * * *'
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
packages: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
publish-gitee:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: ${{ github.event.workflow_run.conclusion == 'success' }}
|
||||||
|
steps:
|
||||||
|
- name: Checkout Code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
lfs: true
|
||||||
|
|
||||||
|
- name: publish_to_gitee
|
||||||
|
id: publish_to_gitee
|
||||||
|
run: |
|
||||||
|
export GITEE_TOKEN=${{ secrets.GITEE_TOKEN }}
|
||||||
|
rm -rf ./pnpm*.yaml
|
||||||
|
npm install -g pnpm
|
||||||
|
pnpm install
|
||||||
|
npm run publish_to_gitee
|
||||||
|
working-directory: ./
|
||||||
|
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
name: publish-github
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: ['v2-dev']
|
||||||
|
paths:
|
||||||
|
- "trigger/publish.trigger"
|
||||||
|
workflow_run:
|
||||||
|
workflows: [ "build-image-for-release" ]
|
||||||
|
types:
|
||||||
|
- completed
|
||||||
|
|
||||||
|
# schedule:
|
||||||
|
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
||||||
|
# - cron: '17 19 * * *'
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
packages: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
publish-github:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: ${{ github.event.workflow_run.conclusion == 'success' }}
|
||||||
|
steps:
|
||||||
|
- name: Checkout Code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
lfs: true
|
||||||
|
|
||||||
|
- name: publish_to_github
|
||||||
|
id: publish_to_github
|
||||||
|
run: |
|
||||||
|
export GITHUB_TOKEN=${{ secrets.GH_TOKEN }}
|
||||||
|
rm -rf ./pnpm*.yaml
|
||||||
|
npm install -g pnpm
|
||||||
|
pnpm install
|
||||||
|
npm run publish_to_github
|
||||||
|
working-directory: ./
|
||||||
|
|
||||||
@@ -1,9 +1,13 @@
|
|||||||
name: build-image-for-test
|
name: build-image-for-release
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches: ['v2-dev']
|
branches: ['v2-dev']
|
||||||
paths:
|
paths:
|
||||||
- "build-dev.trigger"
|
- "trigger/release.trigger"
|
||||||
|
# workflow_run:
|
||||||
|
# workflows: [ "deploy-demo" ]
|
||||||
|
# types:
|
||||||
|
# - completed
|
||||||
|
|
||||||
# schedule:
|
# schedule:
|
||||||
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
||||||
@@ -20,7 +24,7 @@ jobs:
|
|||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
ref: v2-dev
|
lfs: true
|
||||||
|
|
||||||
- name: get_certd_version
|
- name: get_certd_version
|
||||||
id: get_certd_version
|
id: get_certd_version
|
||||||
@@ -75,17 +79,19 @@ jobs:
|
|||||||
username: ${{ secrets.dockerhub_username }}
|
username: ${{ secrets.dockerhub_username }}
|
||||||
password: ${{ secrets.dockerhub_password }}
|
password: ${{ secrets.dockerhub_password }}
|
||||||
|
|
||||||
# - name: Build default platforms
|
- name: Build default platforms
|
||||||
# uses: docker/build-push-action@v6
|
uses: docker/build-push-action@v6
|
||||||
# with:
|
with:
|
||||||
# platforms: linux/amd64,linux/arm64
|
platforms: linux/amd64,linux/arm64
|
||||||
# push: true
|
push: true
|
||||||
# context: ./packages/ui/
|
context: ./packages/ui/
|
||||||
# tags: |
|
tags: |
|
||||||
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-dev:latest
|
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest
|
||||||
# greper/certd-dev:latest
|
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}
|
||||||
# ghcr.io/${{ github.repository }}:dev-latest
|
greper/certd:latest
|
||||||
|
greper/certd:${{steps.get_certd_version.outputs.result}}
|
||||||
|
ghcr.io/${{ github.repository }}:latest
|
||||||
|
ghcr.io/${{ github.repository }}:${{steps.get_certd_version.outputs.result}}
|
||||||
- name: Build armv7
|
- name: Build armv7
|
||||||
uses: docker/build-push-action@v6
|
uses: docker/build-push-action@v6
|
||||||
with:
|
with:
|
||||||
@@ -96,4 +102,32 @@ jobs:
|
|||||||
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:armv7
|
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:armv7
|
||||||
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}-armv7
|
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}-armv7
|
||||||
greper/certd:armv7
|
greper/certd:armv7
|
||||||
greper/certd:${{steps.get_certd_version.outputs.result}}-armv7
|
greper/certd:${{steps.get_certd_version.outputs.result}}-armv7
|
||||||
|
ghcr.io/${{ github.repository }}:armv7
|
||||||
|
ghcr.io/${{ github.repository }}:${{steps.get_certd_version.outputs.result}}-armv7
|
||||||
|
|
||||||
|
# - name: Build agent
|
||||||
|
# uses: docker/build-push-action@v6
|
||||||
|
# with:
|
||||||
|
# platforms: linux/amd64,linux/arm64
|
||||||
|
# push: true
|
||||||
|
# context: ./packages/ui/agent/
|
||||||
|
# tags: |
|
||||||
|
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-agent:latest
|
||||||
|
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-agent:${{steps.get_certd_version.outputs.result}}
|
||||||
|
# greper/certd-agent:latest
|
||||||
|
# greper/certd-agent:${{steps.get_certd_version.outputs.result}}
|
||||||
|
- name: deploy-certd-doc
|
||||||
|
uses: tyrrrz/action-http-request@master
|
||||||
|
with:
|
||||||
|
url: ${{ secrets.WEBHOOK_CERTD_DOC }}
|
||||||
|
method: POST
|
||||||
|
body: |
|
||||||
|
{
|
||||||
|
"CERTD_VERSION": "1.0.0"
|
||||||
|
}
|
||||||
|
headers: |
|
||||||
|
Content-Type: application/json
|
||||||
|
retry-count: 3
|
||||||
|
retry-delay: 5000
|
||||||
|
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
name: sync-to-atomgit-dev
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: ['v2-dev']
|
||||||
|
# schedule:
|
||||||
|
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
||||||
|
# - cron: '17 19 * * *'
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
sync:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout work repo # 1. 检出当前仓库(certd-sync-work)
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
lfs: true
|
||||||
|
- name: Set git user # 2. 给git命令设置用户名和邮箱,↙↙↙ 改成你的name和email
|
||||||
|
run: |
|
||||||
|
git config --global user.name "xiaojunnuo"
|
||||||
|
git config --global user.email "xiaojunnuo@qq.com"
|
||||||
|
|
||||||
|
- name: Set git token # 3. 给git命令设置token,用于push到目标仓库
|
||||||
|
uses: de-vri-es/setup-git-credentials@v2
|
||||||
|
with: # token 格式为: username:password
|
||||||
|
credentials: https://greper:${{secrets.ATOMGIT_TOKEN}}@atomgit.com
|
||||||
|
|
||||||
|
- name: push to atomgit # 4. 执行同步
|
||||||
|
run: |
|
||||||
|
git remote add upstream https://atomgit.com/certd/certd
|
||||||
|
git push --set-upstream upstream v2-dev
|
||||||
|
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
name: sync-to-atomgit
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: ['v2']
|
||||||
|
# schedule:
|
||||||
|
# - # 国际时间 19:17 执行,北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
|
||||||
|
# - cron: '17 19 * * *'
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
sync:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout work repo # 1. 检出当前仓库(certd-sync-work)
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
lfs: true
|
||||||
|
- name: Set git user # 2. 给git命令设置用户名和邮箱,↙↙↙ 改成你的name和email
|
||||||
|
run: |
|
||||||
|
git config --global user.name "xiaojunnuo"
|
||||||
|
git config --global user.email "xiaojunnuo@qq.com"
|
||||||
|
|
||||||
|
- name: Set git token # 3. 给git命令设置token,用于push到目标仓库
|
||||||
|
uses: de-vri-es/setup-git-credentials@v2
|
||||||
|
with: # token 格式为: username:password
|
||||||
|
credentials: https://greper:${{secrets.ATOMGIT_TOKEN}}@atomgit.com
|
||||||
|
|
||||||
|
- name: push to atomgit # 4. 执行同步
|
||||||
|
run: |
|
||||||
|
git remote add upstream https://atomgit.com/certd/certd
|
||||||
|
git push --set-upstream upstream v2
|
||||||
|
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
./packages/core/lego
|
./packages/core/lego
|
||||||
# IntelliJ project files
|
# IntelliJ project files
|
||||||
.vscode/
|
|
||||||
node_modules/
|
node_modules/
|
||||||
npm-debug.log
|
npm-debug.log
|
||||||
yarn-error.log
|
yarn-error.log
|
||||||
@@ -17,6 +16,7 @@ gen
|
|||||||
/test/*.private.*
|
/test/*.private.*
|
||||||
|
|
||||||
/*.log
|
/*.log
|
||||||
|
nohup.out
|
||||||
|
|
||||||
/packages/ui/*/.idea
|
/packages/ui/*/.idea
|
||||||
/packages/ui/*/node_modules
|
/packages/ui/*/node_modules
|
||||||
@@ -29,5 +29,6 @@ test/**/*.js
|
|||||||
/packages/ui/certd-server/data/db.sqlite
|
/packages/ui/certd-server/data/db.sqlite
|
||||||
/packages/ui/certd-server/data/keys.yaml
|
/packages/ui/certd-server/data/keys.yaml
|
||||||
/packages/pro/
|
/packages/pro/
|
||||||
|
test.js
|
||||||
test.js
|
.history
|
||||||
|
/logs
|
||||||
@@ -1,2 +1,6 @@
|
|||||||
link-workspace-packages=deep
|
link-workspace-packages=deep
|
||||||
prefer-workspace-packages=true
|
prefer-workspace-packages=true
|
||||||
|
better_sqlite3_binary_host=https://registry.npmmirror.com/-/binary/better-sqlite3
|
||||||
|
better_sqlite3_binary_host_mirror=https://registry.npmmirror.com/-/binary/better-sqlite3
|
||||||
|
better-sqlite3_binary_host=https://registry.npmmirror.com/-/binary/better-sqlite3
|
||||||
|
better-sqlite3_binary_host_mirror=https://registry.npmmirror.com/-/binary/better-sqlite3
|
||||||
@@ -0,0 +1,301 @@
|
|||||||
|
# Access 插件开发技能
|
||||||
|
|
||||||
|
## 什么是 Access 插件
|
||||||
|
|
||||||
|
Access 插件是 Certd 系统中用于存储用户第三方应用授权数据的插件,例如用户名密码、accessSecret 或 accessToken 等。同时,它还负责对接实现第三方的 API 接口,供其他插件调用使用。
|
||||||
|
|
||||||
|
## 开发步骤
|
||||||
|
|
||||||
|
### 1. 导入必要的依赖
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
import { AccessInput, BaseAccess, IsAccess, Pager, PageRes, PageSearch } from '@certd/pipeline';
|
||||||
|
import { DomainRecord } from '@certd/plugin-lib';
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. 使用 @IsAccess 注解注册插件
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
@IsAccess({
|
||||||
|
name: 'demo', // 插件唯一标识
|
||||||
|
title: '授权插件示例', // 插件标题
|
||||||
|
icon: 'clarity:plugin-line', // 插件图标
|
||||||
|
desc: '这是一个示例授权插件,用于演示如何实现一个授权插件', // 插件描述
|
||||||
|
})
|
||||||
|
export class DemoAccess extends BaseAccess {
|
||||||
|
// 插件实现...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. 定义授权属性
|
||||||
|
|
||||||
|
使用 `@AccessInput` 注解定义授权属性:
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
@AccessInput({
|
||||||
|
title: '授权方式',
|
||||||
|
value: 'apiKey', // 默认值
|
||||||
|
component: {
|
||||||
|
name: "a-select", // 基于 antdv 的输入组件
|
||||||
|
vModel: "value", // v-model 绑定的属性名
|
||||||
|
options: [ // 组件参数
|
||||||
|
{ label: "API密钥(推荐)", value: "apiKey" },
|
||||||
|
{ label: "账号密码", value: "account" },
|
||||||
|
],
|
||||||
|
placeholder: 'demoKeyId',
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
apiType = '';
|
||||||
|
|
||||||
|
@AccessInput({
|
||||||
|
title: '密钥Id',
|
||||||
|
component: {
|
||||||
|
name:"a-input",
|
||||||
|
allowClear: true,
|
||||||
|
placeholder: 'demoKeyId',
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
demoKeyId = '';
|
||||||
|
|
||||||
|
@AccessInput({
|
||||||
|
title: '密钥',//标题
|
||||||
|
required: true, //text组件可以省略
|
||||||
|
encrypt: true, //该属性是否需要加密
|
||||||
|
})
|
||||||
|
demoKeySecret = '';
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. 实现测试方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
@AccessInput({
|
||||||
|
title: "测试",
|
||||||
|
component: {
|
||||||
|
name: "api-test",
|
||||||
|
action: "TestRequest"
|
||||||
|
},
|
||||||
|
helper: "点击测试接口是否正常"
|
||||||
|
})
|
||||||
|
testRequest = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 会通过上面的testRequest参数在ui界面上生成测试按钮,供用户测试接口调用是否正常
|
||||||
|
*/
|
||||||
|
async onTestRequest() {
|
||||||
|
await this.GetDomainList({});
|
||||||
|
return "ok"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 5. 实现 API 方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
/**
|
||||||
|
* 获api接口示例 取域名列表,
|
||||||
|
*/
|
||||||
|
async GetDomainList(req: PageSearch): Promise<PageRes<DomainRecord>> {
|
||||||
|
//输出日志必须使用ctx.logger
|
||||||
|
this.ctx.logger.info(`获取域名列表,req:${JSON.stringify(req)}`);
|
||||||
|
const pager = new Pager(req);
|
||||||
|
const resp = await this.doRequest({
|
||||||
|
action: "ListDomains",
|
||||||
|
data: {
|
||||||
|
domain: req.searchKey,
|
||||||
|
offset: pager.getOffset(),
|
||||||
|
limit: pager.pageSize,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const total = resp?.TotalCount || 0;
|
||||||
|
let list = resp?.DomainList?.map((item) => {
|
||||||
|
item.domain = item.Domain;
|
||||||
|
item.id = item.DomainId;
|
||||||
|
return item;
|
||||||
|
})
|
||||||
|
return {
|
||||||
|
total,
|
||||||
|
list
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 通用api调用方法, 具体如何构造请求体,需参考对应应用的API文档
|
||||||
|
*/
|
||||||
|
async doRequest(req: { action: string, data?: any }) {
|
||||||
|
/**
|
||||||
|
this.ctx中包含很多有用的工具类
|
||||||
|
type AccessContext = {
|
||||||
|
http: HttpClient;
|
||||||
|
logger: ILogger;
|
||||||
|
utils: typeof utils;
|
||||||
|
accessService: IAccessService;
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
const res = await this.ctx.http.request({
|
||||||
|
url: "https://api.demo.cn/api/",
|
||||||
|
method: "POST",
|
||||||
|
data: {
|
||||||
|
Action: req.action,
|
||||||
|
Body: req.data
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (res.Code !== 0) {
|
||||||
|
//异常处理
|
||||||
|
throw new Error(res.Message || "请求失败");
|
||||||
|
}
|
||||||
|
return res.Resp;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 注意事项
|
||||||
|
|
||||||
|
1. **插件命名**:插件名称应简洁明了,反映其功能。
|
||||||
|
2. **属性加密**:对于敏感信息(如密钥),应设置 `encrypt: true`。
|
||||||
|
3. **日志输出**:必须使用 `this.ctx.logger` 输出日志,而不是 `console`。
|
||||||
|
4. **错误处理**:API 调用失败时应抛出明确的错误信息。
|
||||||
|
5. **测试方法**:实现 `onTestRequest` 方法,以便用户可以测试授权是否正常。
|
||||||
|
|
||||||
|
## 完整示例
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
import { AccessInput, BaseAccess, IsAccess, Pager, PageRes, PageSearch } from '@certd/pipeline';
|
||||||
|
import { DomainRecord } from '@certd/plugin-lib';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 这个注解将注册一个授权配置
|
||||||
|
* 在certd的后台管理系统中,用户可以选择添加此类型的授权
|
||||||
|
*/
|
||||||
|
@IsAccess({
|
||||||
|
name: 'demo',
|
||||||
|
title: '授权插件示例',
|
||||||
|
icon: 'clarity:plugin-line', //插件图标
|
||||||
|
desc: '这是一个示例授权插件,用于演示如何实现一个授权插件',
|
||||||
|
})
|
||||||
|
export class DemoAccess extends BaseAccess {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 授权属性配置
|
||||||
|
*/
|
||||||
|
@AccessInput({
|
||||||
|
title: '授权方式',
|
||||||
|
value: 'apiKey', //默认值
|
||||||
|
component: {
|
||||||
|
name: "a-select", //基于antdv的输入组件
|
||||||
|
vModel: "value", // v-model绑定的属性名
|
||||||
|
options: [ //组件参数
|
||||||
|
{
|
||||||
|
label: "API密钥(推荐)",
|
||||||
|
value: "apiKey"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "账号密码",
|
||||||
|
value: "account"
|
||||||
|
},
|
||||||
|
],
|
||||||
|
placeholder: 'demoKeyId',
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
apiType = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 授权属性配置
|
||||||
|
*/
|
||||||
|
@AccessInput({
|
||||||
|
title: '密钥Id',
|
||||||
|
component: {
|
||||||
|
name:"a-input",
|
||||||
|
allowClear: true,
|
||||||
|
placeholder: 'demoKeyId',
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
demoKeyId = '';
|
||||||
|
|
||||||
|
@AccessInput({
|
||||||
|
title: '密钥',//标题
|
||||||
|
required: true, //text组件可以省略
|
||||||
|
encrypt: true, //该属性是否需要加密
|
||||||
|
})
|
||||||
|
demoKeySecret = '';
|
||||||
|
|
||||||
|
|
||||||
|
@AccessInput({
|
||||||
|
title: "测试",
|
||||||
|
component: {
|
||||||
|
name: "api-test",
|
||||||
|
action: "TestRequest"
|
||||||
|
},
|
||||||
|
helper: "点击测试接口是否正常"
|
||||||
|
})
|
||||||
|
testRequest = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 会通过上面的testRequest参数在ui界面上生成测试按钮,供用户测试接口调用是否正常
|
||||||
|
*/
|
||||||
|
async onTestRequest() {
|
||||||
|
await this.GetDomainList({});
|
||||||
|
return "ok"
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获api接口示例 取域名列表,
|
||||||
|
*/
|
||||||
|
async GetDomainList(req: PageSearch): Promise<PageRes<DomainRecord>> {
|
||||||
|
//输出日志必须使用ctx.logger
|
||||||
|
this.ctx.logger.info(`获取域名列表,req:${JSON.stringify(req)}`);
|
||||||
|
const pager = new Pager(req);
|
||||||
|
const resp = await this.doRequest({
|
||||||
|
action: "ListDomains",
|
||||||
|
data: {
|
||||||
|
domain: req.searchKey,
|
||||||
|
offset: pager.getOffset(),
|
||||||
|
limit: pager.pageSize,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const total = resp?.TotalCount || 0;
|
||||||
|
let list = resp?.DomainList?.map((item) => {
|
||||||
|
item.domain = item.Domain;
|
||||||
|
item.id = item.DomainId;
|
||||||
|
return item;
|
||||||
|
})
|
||||||
|
return {
|
||||||
|
total,
|
||||||
|
list
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// 还可以继续编写API
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 通用api调用方法, 具体如何构造请求体,需参考对应应用的API文档
|
||||||
|
*/
|
||||||
|
async doRequest(req: { action: string, data?: any }) {
|
||||||
|
/**
|
||||||
|
this.ctx中包含很多有用的工具类
|
||||||
|
type AccessContext = {
|
||||||
|
http: HttpClient;
|
||||||
|
logger: ILogger;
|
||||||
|
utils: typeof utils;
|
||||||
|
accessService: IAccessService;
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
const res = await this.ctx.http.request({
|
||||||
|
url: "https://api.demo.cn/api/",
|
||||||
|
method: "POST",
|
||||||
|
data: {
|
||||||
|
Action: req.action,
|
||||||
|
Body: req.data
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (res.Code !== 0) {
|
||||||
|
//异常处理
|
||||||
|
throw new Error(res.Message || "请求失败");
|
||||||
|
}
|
||||||
|
return res.Resp;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
我需要开发一个 Access 插件,用于存储和管理第三方应用的授权信息。请指导我如何实现。
|
||||||
@@ -0,0 +1,145 @@
|
|||||||
|
# Access 插件开发指南
|
||||||
|
|
||||||
|
## 开发步骤
|
||||||
|
|
||||||
|
### 1. 导入必要的依赖
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
import { AccessInput, BaseAccess, IsAccess, Pager, PageRes, PageSearch } from '@certd/pipeline';
|
||||||
|
import { DomainRecord } from '@certd/plugin-lib';
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. 使用 @IsAccess 注解注册插件
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
@IsAccess({
|
||||||
|
name: 'demo', // 插件唯一标识
|
||||||
|
title: '授权插件示例', // 插件标题
|
||||||
|
icon: 'clarity:plugin-line', // 插件图标
|
||||||
|
desc: '这是一个示例授权插件,用于演示如何实现一个授权插件', // 插件描述
|
||||||
|
})
|
||||||
|
export class DemoAccess extends BaseAccess {
|
||||||
|
// 插件实现...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. 定义授权属性
|
||||||
|
|
||||||
|
使用 `@AccessInput` 注解定义授权属性:
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
@AccessInput({
|
||||||
|
title: '授权方式',
|
||||||
|
value: 'apiKey', // 默认值
|
||||||
|
component: {
|
||||||
|
name: "a-select", // 基于 antdv 的输入组件
|
||||||
|
vModel: "value", // v-model 绑定的属性名
|
||||||
|
options: [ // 组件参数
|
||||||
|
{ label: "API密钥(推荐)", value: "apiKey" },
|
||||||
|
{ label: "账号密码", value: "account" },
|
||||||
|
],
|
||||||
|
placeholder: 'demoKeyId',
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
apiType = '';
|
||||||
|
|
||||||
|
@AccessInput({
|
||||||
|
title: '密钥Id',
|
||||||
|
component: {
|
||||||
|
name:"a-input",
|
||||||
|
allowClear: true,
|
||||||
|
placeholder: 'demoKeyId',
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
demoKeyId = '';
|
||||||
|
|
||||||
|
@AccessInput({
|
||||||
|
title: '密钥',//标题
|
||||||
|
required: true, //text组件可以省略
|
||||||
|
encrypt: true, //该属性是否需要加密
|
||||||
|
})
|
||||||
|
demoKeySecret = '';
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. 实现测试方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
@AccessInput({
|
||||||
|
title: "测试",
|
||||||
|
component: {
|
||||||
|
name: "api-test",
|
||||||
|
action: "TestRequest"
|
||||||
|
},
|
||||||
|
helper: "点击测试接口是否正常"
|
||||||
|
})
|
||||||
|
testRequest = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 会通过上面的testRequest参数在ui界面上生成测试按钮,供用户测试接口调用是否正常
|
||||||
|
*/
|
||||||
|
async onTestRequest() {
|
||||||
|
await this.GetDomainList({});
|
||||||
|
return "ok"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 5. 实现 API 方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
/**
|
||||||
|
* 获api接口示例 取域名列表,
|
||||||
|
*/
|
||||||
|
async GetDomainList(req: PageSearch): Promise<PageRes<DomainRecord>> {
|
||||||
|
//输出日志必须使用ctx.logger
|
||||||
|
this.ctx.logger.info(`获取域名列表,req:${JSON.stringify(req)}`);
|
||||||
|
const pager = new Pager(req);
|
||||||
|
const resp = await this.doRequest({
|
||||||
|
action: "ListDomains",
|
||||||
|
data: {
|
||||||
|
domain: req.searchKey,
|
||||||
|
offset: pager.getOffset(),
|
||||||
|
limit: pager.pageSize,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const total = resp?.TotalCount || 0;
|
||||||
|
let list = resp?.DomainList?.map((item) => {
|
||||||
|
item.domain = item.Domain;
|
||||||
|
item.id = item.DomainId;
|
||||||
|
return item;
|
||||||
|
})
|
||||||
|
return {
|
||||||
|
total,
|
||||||
|
list
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 通用api调用方法, 具体如何构造请求体,需参考对应应用的API文档
|
||||||
|
*/
|
||||||
|
async doRequest(req: { action: string, data?: any }) {
|
||||||
|
const res = await this.ctx.http.request({
|
||||||
|
url: "https://api.demo.cn/api/",
|
||||||
|
method: "POST",
|
||||||
|
data: {
|
||||||
|
Action: req.action,
|
||||||
|
Body: req.data
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (res.Code !== 0) {
|
||||||
|
//异常处理
|
||||||
|
throw new Error(res.Message || "请求失败");
|
||||||
|
}
|
||||||
|
return res.Resp;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 注意事项
|
||||||
|
|
||||||
|
1. **插件命名**:插件名称应简洁明了,反映其功能。
|
||||||
|
2. **属性加密**:对于敏感信息(如密钥),应设置 `encrypt: true`。
|
||||||
|
3. **日志输出**:必须使用 `this.ctx.logger` 输出日志,而不是 `console`。
|
||||||
|
4. **错误处理**:API 调用失败时应抛出明确的错误信息。
|
||||||
|
5. **测试方法**:实现 `onTestRequest` 方法,以便用户可以测试授权是否正常。
|
||||||
@@ -0,0 +1,212 @@
|
|||||||
|
# DNS Provider 插件开发技能
|
||||||
|
|
||||||
|
## 什么是 DNS Provider 插件
|
||||||
|
|
||||||
|
DNS Provider 插件是 Certd 系统中的 DNS 提供商插件,它用于在 ACME 申请证书时给域名添加 TXT 解析记录,以验证域名所有权。
|
||||||
|
|
||||||
|
## 开发步骤
|
||||||
|
|
||||||
|
### 1. 导入必要的依赖
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
import { AbstractDnsProvider, CreateRecordOptions, IsDnsProvider, RemoveRecordOptions } from '@certd/plugin-cert';
|
||||||
|
import { DemoAccess } from './access.js';
|
||||||
|
import { isDev } from '../../utils/env.js';
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. 定义记录数据结构
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
type DemoRecord = {
|
||||||
|
// 这里定义 Record 记录的数据结构,跟对应云平台接口返回值一样即可,一般是拿到 id 就行,用于删除 txt 解析记录,清理申请痕迹
|
||||||
|
// id:string
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. 使用 @IsDnsProvider 注解注册插件
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// 这里通过 IsDnsProvider 注册一个 dnsProvider
|
||||||
|
@IsDnsProvider({
|
||||||
|
name: 'demo', // 插件唯一标识
|
||||||
|
title: 'Dns提供商Demo', // 插件标题
|
||||||
|
desc: 'dns provider示例', // 插件描述
|
||||||
|
icon: 'clarity:plugin-line', // 插件图标
|
||||||
|
// 这里是对应的云平台的 access 类型名称
|
||||||
|
accessType: 'demo',
|
||||||
|
order: 99, // 排序
|
||||||
|
})
|
||||||
|
export class DemoDnsProvider extends AbstractDnsProvider<DemoRecord> {
|
||||||
|
access!: DemoAccess;
|
||||||
|
|
||||||
|
async onInstance() {
|
||||||
|
this.access = this.ctx.access as DemoAccess;
|
||||||
|
// 也可以通过 ctx 成员变量传递 context
|
||||||
|
this.logger.debug('access', this.access);
|
||||||
|
// 初始化的操作
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
|
||||||
|
// 插件实现...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. 实现 createRecord 方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
/**
|
||||||
|
* 创建 dns 解析记录,用于验证域名所有权
|
||||||
|
*/
|
||||||
|
async createRecord(options: CreateRecordOptions): Promise<any> {
|
||||||
|
/**
|
||||||
|
* options 参数说明
|
||||||
|
* fullRecord: '_acme-challenge.example.com',
|
||||||
|
* value: 一串 uuid
|
||||||
|
* type: 'TXT',
|
||||||
|
* domain: 'example.com'
|
||||||
|
*/
|
||||||
|
const { fullRecord, value, type, domain } = options;
|
||||||
|
this.logger.info('添加域名解析:', fullRecord, value, type, domain);
|
||||||
|
|
||||||
|
// 调用创建 dns 解析记录的对应的云端接口,创建 txt 类型的 dns 解析记录
|
||||||
|
// 请根据实际接口情况调用,例如:
|
||||||
|
// const createDnsRecordUrl = "xxx"
|
||||||
|
// const record = this.http.post(createDnsRecordUrl,{
|
||||||
|
// // 授权参数
|
||||||
|
// // 创建 dns 解析记录的参数
|
||||||
|
// })
|
||||||
|
// // 返回本次创建的 dns 解析记录,这个记录会在删除的时候用到
|
||||||
|
// return record
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 5. 实现 removeRecord 方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
/**
|
||||||
|
* 删除 dns 解析记录,清理申请痕迹
|
||||||
|
* @param options
|
||||||
|
*/
|
||||||
|
async removeRecord(options: RemoveRecordOptions<DemoRecord>): Promise<void> {
|
||||||
|
const { fullRecord, value, domain } = options.recordReq;
|
||||||
|
const record = options.recordRes;
|
||||||
|
this.logger.info('删除域名解析:', domain, fullRecord, value, record);
|
||||||
|
// 这里调用删除 txt dns 解析记录接口
|
||||||
|
// 请根据实际接口情况调用,例如:
|
||||||
|
|
||||||
|
// const deleteDnsRecordUrl = "xxx"
|
||||||
|
// const res = this.http.delete(deleteDnsRecordUrl,{
|
||||||
|
// // 授权参数
|
||||||
|
// // 删除 dns 解析记录的参数
|
||||||
|
// })
|
||||||
|
|
||||||
|
|
||||||
|
this.logger.info('删除域名解析成功:', fullRecord, value);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 6. 实例化插件
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// 实例化这个 provider,将其自动注册到系统中
|
||||||
|
if (isDev()) {
|
||||||
|
// 你的实现 要去掉这个 if,不然生产环境将不会显示
|
||||||
|
new DemoDnsProvider();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 注意事项
|
||||||
|
|
||||||
|
1. **插件命名**:插件名称应简洁明了,反映其功能。
|
||||||
|
2. **accessType**:必须指定对应的云平台的 access 类型名称。
|
||||||
|
3. **记录结构**:定义适合对应云平台的记录数据结构,至少包含 id 字段用于删除记录。
|
||||||
|
4. **日志输出**:使用 `this.logger` 输出日志,而不是 `console`。
|
||||||
|
5. **错误处理**:API 调用失败时应抛出明确的错误信息。
|
||||||
|
6. **实例化**:生产环境中应移除 `if (isDev())` 条件,确保插件在生产环境中也能被注册。
|
||||||
|
|
||||||
|
## 完整示例
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
import { AbstractDnsProvider, CreateRecordOptions, IsDnsProvider, RemoveRecordOptions } from '@certd/plugin-cert';
|
||||||
|
import { DemoAccess } from './access.js';
|
||||||
|
import { isDev } from '../../utils/env.js';
|
||||||
|
|
||||||
|
type DemoRecord = {
|
||||||
|
// 这里定义 Record 记录的数据结构,跟对应云平台接口返回值一样即可,一般是拿到 id 就行,用于删除 txt 解析记录,清理申请痕迹
|
||||||
|
// id:string
|
||||||
|
};
|
||||||
|
|
||||||
|
// 这里通过 IsDnsProvider 注册一个 dnsProvider
|
||||||
|
@IsDnsProvider({
|
||||||
|
name: 'demo',
|
||||||
|
title: 'Dns提供商Demo',
|
||||||
|
desc: 'dns provider示例',
|
||||||
|
icon: 'clarity:plugin-line',
|
||||||
|
// 这里是对应的云平台的 access 类型名称
|
||||||
|
accessType: 'demo',
|
||||||
|
order: 99,
|
||||||
|
})
|
||||||
|
export class DemoDnsProvider extends AbstractDnsProvider<DemoRecord> {
|
||||||
|
access!: DemoAccess;
|
||||||
|
|
||||||
|
async onInstance() {
|
||||||
|
this.access = this.ctx.access as DemoAccess;
|
||||||
|
// 也可以通过 ctx 成员变量传递 context
|
||||||
|
this.logger.debug('access', this.access);
|
||||||
|
// 初始化的操作
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 创建 dns 解析记录,用于验证域名所有权
|
||||||
|
*/
|
||||||
|
async createRecord(options: CreateRecordOptions): Promise<any> {
|
||||||
|
/**
|
||||||
|
* options 参数说明
|
||||||
|
* fullRecord: '_acme-challenge.example.com',
|
||||||
|
* value: 一串 uuid
|
||||||
|
* type: 'TXT',
|
||||||
|
* domain: 'example.com'
|
||||||
|
*/
|
||||||
|
const { fullRecord, value, type, domain } = options;
|
||||||
|
this.logger.info('添加域名解析:', fullRecord, value, type, domain);
|
||||||
|
|
||||||
|
// 调用创建 dns 解析记录的对应的云端接口,创建 txt 类型的 dns 解析记录
|
||||||
|
// 请根据实际接口情况调用,例如:
|
||||||
|
// const createDnsRecordUrl = "xxx"
|
||||||
|
// const record = this.http.post(createDnsRecordUrl,{
|
||||||
|
// // 授权参数
|
||||||
|
// // 创建 dns 解析记录的参数
|
||||||
|
// })
|
||||||
|
// // 返回本次创建的 dns 解析记录,这个记录会在删除的时候用到
|
||||||
|
// return record
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除 dns 解析记录,清理申请痕迹
|
||||||
|
* @param options
|
||||||
|
*/
|
||||||
|
async removeRecord(options: RemoveRecordOptions<DemoRecord>): Promise<void> {
|
||||||
|
const { fullRecord, value, domain } = options.recordReq;
|
||||||
|
const record = options.recordRes;
|
||||||
|
this.logger.info('删除域名解析:', domain, fullRecord, value, record);
|
||||||
|
// 这里调用删除 txt dns 解析记录接口
|
||||||
|
// 请根据实际接口情况调用,例如:
|
||||||
|
|
||||||
|
// const deleteDnsRecordUrl = "xxx"
|
||||||
|
// const res = this.http.delete(deleteDnsRecordUrl,{
|
||||||
|
// // 授权参数
|
||||||
|
// // 删除 dns 解析记录的参数
|
||||||
|
// })
|
||||||
|
|
||||||
|
|
||||||
|
this.logger.info('删除域名解析成功:', fullRecord, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 实例化这个 provider,将其自动注册到系统中
|
||||||
|
if (isDev()) {
|
||||||
|
// 你的实现 要去掉这个 if,不然生产环境将不会显示
|
||||||
|
new DemoDnsProvider();
|
||||||
|
}
|
||||||
|
```
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
我需要开发一个 DNS Provider 插件,用于在 ACME 申请证书时添加 TXT 解析记录。请指导我如何实现。
|
||||||
@@ -0,0 +1,121 @@
|
|||||||
|
# DNS Provider 插件开发指南
|
||||||
|
|
||||||
|
## 开发步骤
|
||||||
|
|
||||||
|
### 1. 导入必要的依赖
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
import { AbstractDnsProvider, CreateRecordOptions, IsDnsProvider, RemoveRecordOptions } from '@certd/plugin-cert';
|
||||||
|
import { DemoAccess } from './access.js';
|
||||||
|
import { isDev } from '../../utils/env.js';
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. 定义记录数据结构
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
type DemoRecord = {
|
||||||
|
// 这里定义 Record 记录的数据结构,跟对应云平台接口返回值一样即可,一般是拿到 id 就行,用于删除 txt 解析记录,清理申请痕迹
|
||||||
|
// id:string
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. 使用 @IsDnsProvider 注解注册插件
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// 这里通过 IsDnsProvider 注册一个 dnsProvider
|
||||||
|
@IsDnsProvider({
|
||||||
|
name: 'demo', // 插件唯一标识
|
||||||
|
title: 'Dns提供商Demo', // 插件标题
|
||||||
|
desc: 'dns provider示例', // 插件描述
|
||||||
|
icon: 'clarity:plugin-line', // 插件图标
|
||||||
|
// 这里是对应的云平台的 access 类型名称
|
||||||
|
accessType: 'demo',
|
||||||
|
order: 99, // 排序
|
||||||
|
})
|
||||||
|
export class DemoDnsProvider extends AbstractDnsProvider<DemoRecord> {
|
||||||
|
access!: DemoAccess;
|
||||||
|
|
||||||
|
async onInstance() {
|
||||||
|
this.access = this.ctx.access as DemoAccess;
|
||||||
|
// 也可以通过 ctx 成员变量传递 context
|
||||||
|
this.logger.debug('access', this.access);
|
||||||
|
// 初始化的操作
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
|
||||||
|
// 插件实现...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. 实现 createRecord 方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
/**
|
||||||
|
* 创建 dns 解析记录,用于验证域名所有权
|
||||||
|
*/
|
||||||
|
async createRecord(options: CreateRecordOptions): Promise<any> {
|
||||||
|
/**
|
||||||
|
* options 参数说明
|
||||||
|
* fullRecord: '_acme-challenge.example.com',
|
||||||
|
* value: 一串 uuid
|
||||||
|
* type: 'TXT',
|
||||||
|
* domain: 'example.com'
|
||||||
|
*/
|
||||||
|
const { fullRecord, value, type, domain } = options;
|
||||||
|
this.logger.info('添加域名解析:', fullRecord, value, type, domain);
|
||||||
|
|
||||||
|
// 调用创建 dns 解析记录的对应的云端接口,创建 txt 类型的 dns 解析记录
|
||||||
|
// 请根据实际接口情况调用,例如:
|
||||||
|
// const createDnsRecordUrl = "xxx"
|
||||||
|
// const record = this.http.post(createDnsRecordUrl,{
|
||||||
|
// // 授权参数
|
||||||
|
// // 创建 dns 解析记录的参数
|
||||||
|
// })
|
||||||
|
// // 返回本次创建的 dns 解析记录,这个记录会在删除的时候用到
|
||||||
|
// return record
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 5. 实现 removeRecord 方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
/**
|
||||||
|
* 删除 dns 解析记录,清理申请痕迹
|
||||||
|
* @param options
|
||||||
|
*/
|
||||||
|
async removeRecord(options: RemoveRecordOptions<DemoRecord>): Promise<void> {
|
||||||
|
const { fullRecord, value, domain } = options.recordReq;
|
||||||
|
const record = options.recordRes;
|
||||||
|
this.logger.info('删除域名解析:', domain, fullRecord, value, record);
|
||||||
|
// 这里调用删除 txt dns 解析记录接口
|
||||||
|
// 请根据实际接口情况调用,例如:
|
||||||
|
|
||||||
|
// const deleteDnsRecordUrl = "xxx"
|
||||||
|
// const res = this.http.delete(deleteDnsRecordUrl,{
|
||||||
|
// // 授权参数
|
||||||
|
// // 删除 dns 解析记录的参数
|
||||||
|
// })
|
||||||
|
|
||||||
|
|
||||||
|
this.logger.info('删除域名解析成功:', fullRecord, value);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 6. 实例化插件
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// 实例化这个 provider,将其自动注册到系统中
|
||||||
|
if (isDev()) {
|
||||||
|
// 你的实现 要去掉这个 if,不然生产环境将不会显示
|
||||||
|
new DemoDnsProvider();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 注意事项
|
||||||
|
|
||||||
|
1. **插件命名**:插件名称应简洁明了,反映其功能。
|
||||||
|
2. **accessType**:必须指定对应的云平台的 access 类型名称。
|
||||||
|
3. **记录结构**:定义适合对应云平台的记录数据结构,至少包含 id 字段用于删除记录。
|
||||||
|
4. **日志输出**:使用 `this.logger` 输出日志,而不是 `console`。
|
||||||
|
5. **错误处理**:API 调用失败时应抛出明确的错误信息。
|
||||||
|
6. **实例化**:生产环境中应移除 `if (isDev())` 条件,确保插件在生产环境中也能被注册。
|
||||||
@@ -0,0 +1,201 @@
|
|||||||
|
# 插件转换工具技能
|
||||||
|
|
||||||
|
## 什么是插件转换工具
|
||||||
|
|
||||||
|
插件转换工具是一个用于将 Certd 插件转换为 YAML 配置文件的命令行工具。它可以分析单个插件文件,识别插件类型,并生成对应的 YAML 配置,方便插件的注册和管理。
|
||||||
|
|
||||||
|
## 工具位置
|
||||||
|
|
||||||
|
`trae/skills/convert-plugin-to-yaml.js`
|
||||||
|
|
||||||
|
## 功能特性
|
||||||
|
|
||||||
|
- **单个插件转换**:支持指定单个插件文件进行转换,而不是扫描整个目录
|
||||||
|
- **自动类型识别**:自动识别插件类型(Access、Task、DNS Provider、Notification、Addon)
|
||||||
|
- **详细日志输出**:提供详细的转换过程日志,便于调试
|
||||||
|
- **YAML 配置生成**:生成标准的 YAML 配置文件
|
||||||
|
- **配置文件保存**:自动将生成的配置保存到 `./metadata` 目录
|
||||||
|
- **可复用函数**:导出了可复用的函数,便于其他模块调用
|
||||||
|
|
||||||
|
## 使用方法
|
||||||
|
|
||||||
|
### 基本用法
|
||||||
|
|
||||||
|
```bash
|
||||||
|
node trae/skills/convert-plugin-to-yaml.js <插件文件路径>
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# 转换 Access 插件
|
||||||
|
node trae/skills/convert-plugin-to-yaml.js packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
|
||||||
|
# 转换 Task 插件
|
||||||
|
node trae/skills/convert-plugin-to-yaml.js packages/ui/certd-server/src/plugins/plugin-demo/plugins/plugin-test.js
|
||||||
|
|
||||||
|
# 转换 DNS Provider 插件
|
||||||
|
node trae/skills/convert-plugin-to-yaml.js packages/ui/certd-server/src/plugins/plugin-demo/dns-provider.js
|
||||||
|
```
|
||||||
|
|
||||||
|
## 转换过程
|
||||||
|
|
||||||
|
1. **加载插件模块**:使用 `import()` 动态加载指定的插件文件
|
||||||
|
2. **分析插件定义**:检查模块导出的对象,寻找带有 `define` 属性的插件
|
||||||
|
3. **识别插件类型**:根据插件的继承关系或属性识别插件类型
|
||||||
|
4. **生成 YAML 配置**:基于插件定义生成标准的 YAML 配置
|
||||||
|
5. **保存配置文件**:将生成的配置保存到 `./metadata` 目录
|
||||||
|
|
||||||
|
## 输出说明
|
||||||
|
|
||||||
|
### 命令行输出
|
||||||
|
|
||||||
|
执行转换命令后,工具会输出以下信息:
|
||||||
|
|
||||||
|
- 插件加载状态
|
||||||
|
- 插件导出的对象列表
|
||||||
|
- 插件类型识别结果
|
||||||
|
- 生成的 YAML 配置内容
|
||||||
|
- 配置文件保存路径
|
||||||
|
|
||||||
|
### 配置文件命名规则
|
||||||
|
|
||||||
|
生成的配置文件命名规则为:
|
||||||
|
|
||||||
|
```
|
||||||
|
<插件类型>[_<子类型>]_<插件名称>.yaml
|
||||||
|
```
|
||||||
|
|
||||||
|
例如:
|
||||||
|
- `access_demo.yaml`(Access 插件)
|
||||||
|
- `deploy_DemoTest.yaml`(Task 插件)
|
||||||
|
- `dnsProvider_demo.yaml`(DNS Provider 插件)
|
||||||
|
|
||||||
|
## 插件类型识别逻辑
|
||||||
|
|
||||||
|
工具通过以下逻辑识别插件类型:
|
||||||
|
|
||||||
|
1. **DNS Provider**:如果插件定义中包含 `accessType` 属性
|
||||||
|
2. **Task**:如果插件继承自 `AbstractTaskPlugin`
|
||||||
|
3. **Notification**:如果插件继承自 `BaseNotification`
|
||||||
|
4. **Access**:如果插件继承自 `BaseAccess`
|
||||||
|
5. **Addon**:如果插件继承自 `BaseAddon`
|
||||||
|
|
||||||
|
## 注意事项
|
||||||
|
|
||||||
|
1. **文件路径**:插件文件路径可以是相对路径或绝对路径
|
||||||
|
2. **文件格式**:仅支持 `.js` 文件,不支持 `.ts` 文件(需要先编译)
|
||||||
|
3. **依赖安装**:执行前确保已安装所有必要的依赖
|
||||||
|
4. **配置目录**:如果 `./metadata` 目录不存在,工具会自动创建
|
||||||
|
5. **错误处理**:如果插件加载失败或识别失败,工具会输出错误信息但不会终止执行
|
||||||
|
|
||||||
|
## 代码结构
|
||||||
|
|
||||||
|
### 主要函数
|
||||||
|
|
||||||
|
1. **isPrototypeOf(value, cls)**:检查对象是否是指定类的原型
|
||||||
|
2. **loadSingleModule(filePath)**:加载单个插件模块
|
||||||
|
3. **convertSinglePlugin(pluginPath)**:分析单个插件并生成 YAML 配置
|
||||||
|
4. **main()**:主函数,处理命令行参数并执行转换
|
||||||
|
|
||||||
|
### 导出函数
|
||||||
|
|
||||||
|
工具导出了以下函数,便于其他模块调用:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
export {
|
||||||
|
convertSinglePlugin, // 转换单个插件
|
||||||
|
loadSingleModule, // 加载单个模块
|
||||||
|
isPrototypeOf // 检查原型关系
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
## 应用场景
|
||||||
|
|
||||||
|
1. **插件开发**:在开发新插件时,快速生成配置文件
|
||||||
|
2. **插件调试**:查看插件的内部定义和配置
|
||||||
|
3. **插件管理**:批量转换现有插件为标准配置格式
|
||||||
|
4. **自动化构建**:集成到构建流程中,自动生成插件配置
|
||||||
|
|
||||||
|
## 示例输出
|
||||||
|
|
||||||
|
### 转换 Access 插件示例
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ node trae/skills/convert-plugin-to-yaml.js packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
开始转换插件: packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
插件模块导出了 1 个对象: DemoAccess
|
||||||
|
处理插件: DemoAccess
|
||||||
|
插件类型: access
|
||||||
|
脚本路径: packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
|
||||||
|
生成的 YAML 配置:
|
||||||
|
name: demo
|
||||||
|
title: 授权插件示例
|
||||||
|
desc: 这是一个示例授权插件,用于演示如何实现一个授权插件
|
||||||
|
icon: clarity:plugin-line
|
||||||
|
pluginType: access
|
||||||
|
type: builtIn
|
||||||
|
scriptFilePath: packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
|
||||||
|
YAML 配置已保存到: ./metadata/access_demo.yaml
|
||||||
|
插件转换完成!
|
||||||
|
```
|
||||||
|
|
||||||
|
### 转换 Task 插件示例
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ node trae/skills/convert-plugin-to-yaml.js packages/ui/certd-server/src/plugins/plugin-demo/plugins/plugin-test.js
|
||||||
|
开始转换插件: packages/ui/certd-server/src/plugins/plugin-demo/plugins/plugin-test.js
|
||||||
|
插件模块导出了 1 个对象: DemoTest
|
||||||
|
处理插件: DemoTest
|
||||||
|
插件类型: deploy
|
||||||
|
脚本路径: packages/ui/certd-server/src/plugins/plugin-demo/plugins/plugin-test.js
|
||||||
|
|
||||||
|
生成的 YAML 配置:
|
||||||
|
name: DemoTest
|
||||||
|
title: Demo-测试插件
|
||||||
|
desc: ""
|
||||||
|
icon: clarity:plugin-line
|
||||||
|
group: other
|
||||||
|
default:
|
||||||
|
strategy:
|
||||||
|
runStrategy: SkipWhenSucceed
|
||||||
|
pluginType: deploy
|
||||||
|
type: builtIn
|
||||||
|
scriptFilePath: packages/ui/certd-server/src/plugins/plugin-demo/plugins/plugin-test.js
|
||||||
|
|
||||||
|
YAML 配置已保存到: ./metadata/deploy_DemoTest.yaml
|
||||||
|
插件转换完成!
|
||||||
|
```
|
||||||
|
|
||||||
|
## 故障排除
|
||||||
|
|
||||||
|
### 常见问题
|
||||||
|
|
||||||
|
1. **模块加载失败**
|
||||||
|
- 原因:插件文件依赖未安装或路径错误
|
||||||
|
- 解决:确保已安装所有依赖,检查文件路径是否正确
|
||||||
|
|
||||||
|
2. **插件类型识别失败**
|
||||||
|
- 原因:插件未正确继承基类或缺少必要的属性
|
||||||
|
- 解决:检查插件代码,确保正确继承对应的基类
|
||||||
|
|
||||||
|
3. **YAML 配置生成失败**
|
||||||
|
- 原因:插件定义格式不正确
|
||||||
|
- 解决:检查插件的 `define` 属性格式是否正确
|
||||||
|
|
||||||
|
4. **配置文件保存失败**
|
||||||
|
- 原因:权限不足或磁盘空间不足
|
||||||
|
- 解决:确保有足够的权限和磁盘空间
|
||||||
|
|
||||||
|
### 调试建议
|
||||||
|
|
||||||
|
- **查看详细日志**:工具会输出详细的转换过程日志,仔细查看日志信息
|
||||||
|
- **检查插件代码**:确保插件代码符合 Certd 插件开发规范
|
||||||
|
- **尝试简化插件**:如果转换失败,尝试创建一个最小化的插件示例进行测试
|
||||||
|
- **检查依赖版本**:确保使用的依赖版本与 Certd 兼容
|
||||||
|
|
||||||
|
## 总结
|
||||||
|
|
||||||
|
插件转换工具是一个方便实用的工具,它可以帮助开发者快速生成插件的 YAML 配置文件,简化插件的注册和管理过程。通过命令行参数指定单个插件文件,工具会自动完成类型识别、配置生成和保存等操作,大大提高了插件开发和管理的效率。
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
我需要将一个插件转换为 YAML 配置文件。请指导我如何使用插件转换工具。
|
||||||
@@ -0,0 +1,95 @@
|
|||||||
|
# 插件转换工具使用指南
|
||||||
|
|
||||||
|
## 工具说明
|
||||||
|
|
||||||
|
插件转换工具用于将单个 Certd 插件转换为 YAML 配置文件,方便插件的注册和管理。
|
||||||
|
|
||||||
|
## 工具位置
|
||||||
|
|
||||||
|
`.trae/skills/plugin-converter/resources/convert-plugin-to-yaml.js`
|
||||||
|
|
||||||
|
## 使用方法
|
||||||
|
|
||||||
|
### 基本用法
|
||||||
|
|
||||||
|
```bash
|
||||||
|
node .trae/skills/plugin-converter/resources/convert-plugin-to-yaml.js <插件文件路径>
|
||||||
|
```
|
||||||
|
|
||||||
|
### 示例
|
||||||
|
|
||||||
|
#### 转换 Access 插件
|
||||||
|
|
||||||
|
```bash
|
||||||
|
node .trae/skills/plugin-converter/resources/convert-plugin-to-yaml.js packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 转换 Task 插件
|
||||||
|
|
||||||
|
```bash
|
||||||
|
node .trae/skills/plugin-converter/resources/convert-plugin-to-yaml.js packages/ui/certd-server/src/plugins/plugin-demo/plugins/plugin-test.js
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 转换 DNS Provider 插件
|
||||||
|
|
||||||
|
```bash
|
||||||
|
node .trae/skills/plugin-converter/resources/convert-plugin-to-yaml.js packages/ui/certd-server/src/plugins/plugin-demo/dns-provider.js
|
||||||
|
```
|
||||||
|
|
||||||
|
## 转换过程
|
||||||
|
|
||||||
|
1. **加载插件模块**:使用 `import()` 动态加载指定的插件文件
|
||||||
|
2. **分析插件定义**:检查模块导出的对象,寻找带有 `define` 属性的插件
|
||||||
|
3. **识别插件类型**:根据插件的继承关系或属性识别插件类型
|
||||||
|
4. **生成 YAML 配置**:基于插件定义生成标准的 YAML 配置
|
||||||
|
5. **保存配置文件**:将生成的配置保存到 `./metadata` 目录
|
||||||
|
|
||||||
|
## 输出说明
|
||||||
|
|
||||||
|
### 命令行输出
|
||||||
|
|
||||||
|
执行转换命令后,工具会输出以下信息:
|
||||||
|
|
||||||
|
- 插件加载状态
|
||||||
|
- 插件导出的对象列表
|
||||||
|
- 插件类型识别结果
|
||||||
|
- 生成的 YAML 配置内容
|
||||||
|
- 配置文件保存路径
|
||||||
|
|
||||||
|
### 配置文件命名规则
|
||||||
|
|
||||||
|
生成的配置文件命名规则为:
|
||||||
|
|
||||||
|
```
|
||||||
|
<插件类型>[_<子类型>]_<插件名称>.yaml
|
||||||
|
```
|
||||||
|
|
||||||
|
例如:
|
||||||
|
- `access_demo.yaml`(Access 插件)
|
||||||
|
- `deploy_DemoTest.yaml`(Task 插件)
|
||||||
|
- `dnsProvider_demo.yaml`(DNS Provider 插件)
|
||||||
|
|
||||||
|
## 示例输出
|
||||||
|
|
||||||
|
### 转换 Access 插件示例
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ node .trae/skills/plugin-converter/resources/convert-plugin-to-yaml.js packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
开始转换插件: packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
插件模块导出了 1 个对象: DemoAccess
|
||||||
|
处理插件: DemoAccess
|
||||||
|
插件类型: access
|
||||||
|
脚本路径: packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
|
||||||
|
生成的 YAML 配置:
|
||||||
|
name: demo
|
||||||
|
title: 授权插件示例
|
||||||
|
desc: 这是一个示例授权插件,用于演示如何实现一个授权插件
|
||||||
|
icon: clarity:plugin-line
|
||||||
|
pluginType: access
|
||||||
|
type: builtIn
|
||||||
|
scriptFilePath: packages/ui/certd-server/src/plugins/plugin-demo/access.js
|
||||||
|
|
||||||
|
YAML 配置已保存到: ./metadata/access_demo.yaml
|
||||||
|
插件转换完成!
|
||||||
|
```
|
||||||
@@ -0,0 +1,160 @@
|
|||||||
|
// 转换单个插件为 YAML 配置的技能脚本
|
||||||
|
|
||||||
|
import path from "path";
|
||||||
|
import fs from "fs";
|
||||||
|
import { pathToFileURL } from "node:url";
|
||||||
|
import * as yaml from "js-yaml";
|
||||||
|
import { AbstractTaskPlugin, BaseAccess, BaseNotification} from "@certd/pipeline";
|
||||||
|
import { BaseAddon} from "@certd/lib-server";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 检查对象是否是指定类的原型
|
||||||
|
*/
|
||||||
|
function isPrototypeOf(value, cls) {
|
||||||
|
return cls.prototype.isPrototypeOf(value.prototype);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 加载单个插件模块
|
||||||
|
*/
|
||||||
|
async function loadSingleModule(filePath) {
|
||||||
|
try {
|
||||||
|
// 转换为 file:// URL(Windows 必需)
|
||||||
|
const moduleUrl = pathToFileURL(filePath).href;
|
||||||
|
const module = await import(moduleUrl);
|
||||||
|
return module.default || module;
|
||||||
|
} catch (err) {
|
||||||
|
console.error(`加载模块 ${filePath} 失败:`, err);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 分析单个插件并生成 YAML 配置
|
||||||
|
*/
|
||||||
|
async function convertSinglePlugin(pluginPath) {
|
||||||
|
console.log(`开始转换插件: ${pluginPath}`);
|
||||||
|
|
||||||
|
// 加载插件模块
|
||||||
|
const module = await loadSingleModule(pluginPath);
|
||||||
|
if (!module) {
|
||||||
|
console.error("加载插件失败,退出");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 处理模块中的所有导出
|
||||||
|
const entry = Object.entries(module);
|
||||||
|
if (entry.length === 0) {
|
||||||
|
console.error("插件模块没有导出任何内容");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`插件模块导出了 ${entry.length} 个对象: ${entry.map(([name]) => name).join(", ")}`);
|
||||||
|
|
||||||
|
// 处理每个导出的对象
|
||||||
|
for (const [name, value] of entry) {
|
||||||
|
// 检查是否是插件(有 define 属性)
|
||||||
|
if (!value.define) {
|
||||||
|
console.log(`跳过非插件对象: ${name}`);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`处理插件: ${name}`);
|
||||||
|
|
||||||
|
// 构建插件定义
|
||||||
|
const pluginDefine = {
|
||||||
|
...value.define
|
||||||
|
};
|
||||||
|
|
||||||
|
let subType = "";
|
||||||
|
|
||||||
|
// 确定插件类型
|
||||||
|
if (pluginDefine.accessType) {
|
||||||
|
pluginDefine.pluginType = "dnsProvider";
|
||||||
|
} else if (isPrototypeOf(value, AbstractTaskPlugin)) {
|
||||||
|
pluginDefine.pluginType = "deploy";
|
||||||
|
} else if (isPrototypeOf(value, BaseNotification)) {
|
||||||
|
pluginDefine.pluginType = "notification";
|
||||||
|
} else if (isPrototypeOf(value, BaseAccess)) {
|
||||||
|
pluginDefine.pluginType = "access";
|
||||||
|
} else if (isPrototypeOf(value, BaseAddon)) {
|
||||||
|
pluginDefine.pluginType = "addon";
|
||||||
|
subType = "_" + (pluginDefine.addonType || "");
|
||||||
|
} else {
|
||||||
|
console.log(`[warning] 未知的插件类型:${pluginDefine.name}`);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
pluginDefine.type = "builtIn";
|
||||||
|
|
||||||
|
// 计算脚本文件路径
|
||||||
|
const relativePath = path.relative(process.cwd(), pluginPath);
|
||||||
|
const scriptFilePath = relativePath.replace(/\\/g, "/").replace(/\.js$/, ".js");
|
||||||
|
pluginDefine.scriptFilePath = scriptFilePath;
|
||||||
|
|
||||||
|
console.log(`插件类型: ${pluginDefine.pluginType}`);
|
||||||
|
console.log(`脚本路径: ${scriptFilePath}`);
|
||||||
|
|
||||||
|
// 生成 YAML 配置
|
||||||
|
const yamlContent = yaml.dump(pluginDefine);
|
||||||
|
console.log("\n生成的 YAML 配置:");
|
||||||
|
console.log(yamlContent);
|
||||||
|
|
||||||
|
// 可选:保存到文件
|
||||||
|
const outputDir = "./metadata";
|
||||||
|
if (!fs.existsSync(outputDir)) {
|
||||||
|
fs.mkdirSync(outputDir, { recursive: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
const outputFileName = `${pluginDefine.pluginType}${subType}_${pluginDefine.name}.yaml`;
|
||||||
|
const outputPath = path.join(outputDir, outputFileName);
|
||||||
|
|
||||||
|
fs.writeFileSync(outputPath, yamlContent, 'utf8');
|
||||||
|
console.log(`\nYAML 配置已保存到: ${outputPath}`);
|
||||||
|
|
||||||
|
return pluginDefine;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.error("未找到有效的插件定义");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 主函数
|
||||||
|
*/
|
||||||
|
async function main() {
|
||||||
|
const args = process.argv.slice(2);
|
||||||
|
|
||||||
|
if (args.length === 0) {
|
||||||
|
console.error("请指定插件文件路径");
|
||||||
|
console.log("用法: node convert-plugin-to-yaml.js <插件文件路径>");
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
const pluginPath = args[0];
|
||||||
|
|
||||||
|
if (!fs.existsSync(pluginPath)) {
|
||||||
|
console.error(`插件文件不存在: ${pluginPath}`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await convertSinglePlugin(pluginPath);
|
||||||
|
console.log("\n插件转换完成!");
|
||||||
|
} catch (error) {
|
||||||
|
console.error("转换过程中出错:", error);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果直接运行此脚本
|
||||||
|
if (import.meta.url === pathToFileURL(process.argv[1]).href) {
|
||||||
|
main();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 导出函数,以便其他模块使用
|
||||||
|
export {
|
||||||
|
convertSinglePlugin,
|
||||||
|
loadSingleModule,
|
||||||
|
isPrototypeOf
|
||||||
|
};
|
||||||
|
|
||||||
@@ -0,0 +1,388 @@
|
|||||||
|
# Task 插件开发技能
|
||||||
|
|
||||||
|
## 什么是 Task 插件
|
||||||
|
|
||||||
|
Task 插件是 Certd 系统中的部署任务插件,它继承自 `AbstractTaskPlugin` 类,被流水线调用 `execute` 方法,将证书部署到对应的应用上。
|
||||||
|
|
||||||
|
## 开发步骤
|
||||||
|
|
||||||
|
### 1. 导入必要的依赖
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
import { AbstractTaskPlugin, IsTaskPlugin, PageSearch, pluginGroups, RunStrategy, TaskInput } from '@certd/pipeline';
|
||||||
|
import { CertInfo, CertReader } from '@certd/plugin-cert';
|
||||||
|
import { createCertDomainGetterInputDefine, createRemoteSelectInputDefine } from '@certd/plugin-lib';
|
||||||
|
import { optionsUtils } from '@certd/basic';
|
||||||
|
import { CertApplyPluginNames} from '@certd/plugin-cert';
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. 使用 @IsTaskPlugin 注解注册插件
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
@IsTaskPlugin({
|
||||||
|
// 命名规范,插件类型+功能,大写字母开头,驼峰命名
|
||||||
|
name: 'DemoTest',
|
||||||
|
title: 'Demo-测试插件', // 插件标题
|
||||||
|
icon: 'clarity:plugin-line', // 插件图标
|
||||||
|
// 插件分组
|
||||||
|
group: pluginGroups.other.key,
|
||||||
|
default: {
|
||||||
|
// 默认值配置照抄即可
|
||||||
|
strategy: {
|
||||||
|
runStrategy: RunStrategy.SkipWhenSucceed,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
// 类名规范,跟上面插件名称(name)一致
|
||||||
|
export class DemoTest extends AbstractTaskPlugin {
|
||||||
|
// 插件实现...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. 定义任务输入参数
|
||||||
|
|
||||||
|
使用 `@TaskInput` 注解定义任务输入参数:
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// 测试参数
|
||||||
|
@TaskInput({
|
||||||
|
title: '属性示例',
|
||||||
|
value: '默认值',
|
||||||
|
component: {
|
||||||
|
// 前端组件配置,具体配置见组件文档 https://www.antdv.com/components/input-cn
|
||||||
|
name: 'a-input',
|
||||||
|
vModel: 'value', // 双向绑定组件的 props 名称
|
||||||
|
},
|
||||||
|
helper: '帮助说明,[链接](https://certd.docmirror.cn)',
|
||||||
|
required: false, // 是否必填
|
||||||
|
})
|
||||||
|
text!: string;
|
||||||
|
|
||||||
|
// 证书选择,此项必须要有
|
||||||
|
@TaskInput({
|
||||||
|
title: '域名证书',
|
||||||
|
helper: '请选择前置任务输出的域名证书',
|
||||||
|
component: {
|
||||||
|
name: 'output-selector',
|
||||||
|
from: [...CertApplyPluginNames],
|
||||||
|
},
|
||||||
|
// required: true, // 必填
|
||||||
|
})
|
||||||
|
cert!: CertInfo;
|
||||||
|
|
||||||
|
@TaskInput(createCertDomainGetterInputDefine({ props: { required: false } }))
|
||||||
|
// 前端可以展示,当前申请的证书域名列表
|
||||||
|
certDomains!: string[];
|
||||||
|
|
||||||
|
// 授权选择框
|
||||||
|
@TaskInput({
|
||||||
|
title: 'demo授权',
|
||||||
|
helper: 'demoAccess授权',
|
||||||
|
component: {
|
||||||
|
name: 'access-selector',
|
||||||
|
type: 'demo', // 固定授权类型
|
||||||
|
},
|
||||||
|
// rules: [{ required: true, message: '此项必填' }],
|
||||||
|
// required: true, // 必填
|
||||||
|
})
|
||||||
|
accessId!: string;
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. 实现插件方法
|
||||||
|
|
||||||
|
#### 4.1 插件实例化时执行的方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// 插件实例化时执行的方法
|
||||||
|
async onInstance() {}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 4.2 插件执行方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// 插件执行方法
|
||||||
|
async execute(): Promise<void> {
|
||||||
|
const { select, text, cert, accessId } = this;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const access = await this.getAccess(accessId);
|
||||||
|
this.logger.debug('access', access);
|
||||||
|
} catch (e) {
|
||||||
|
this.logger.error('获取授权失败', e);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const certReader = new CertReader(cert);
|
||||||
|
this.logger.debug('certReader', certReader);
|
||||||
|
} catch (e) {
|
||||||
|
this.logger.error('读取crt失败', e);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.info('DemoTestPlugin execute');
|
||||||
|
this.logger.info('text:', text);
|
||||||
|
this.logger.info('select:', select);
|
||||||
|
this.logger.info('switch:', this.switch);
|
||||||
|
this.logger.info('授权id:', accessId);
|
||||||
|
|
||||||
|
// 具体的部署逻辑
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 4.3 后端获取选项方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
@TaskInput(
|
||||||
|
createRemoteSelectInputDefine({
|
||||||
|
title: '从后端获取选项',
|
||||||
|
helper: '选择时可以从后端获取选项',
|
||||||
|
action: DemoTest.prototype.onGetSiteList.name,
|
||||||
|
// 当以下参数变化时,触发获取选项
|
||||||
|
watches: ['certDomains', 'accessId'],
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
siteName!: string | string[];
|
||||||
|
|
||||||
|
// 从后端获取选项的方法
|
||||||
|
async onGetSiteList(req: PageSearch) {
|
||||||
|
if (!this.accessId) {
|
||||||
|
throw new Error('请选择Access授权');
|
||||||
|
}
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const access = await this.getAccess(this.accessId);
|
||||||
|
|
||||||
|
// const siteRes = await access.GetDomainList(req);
|
||||||
|
// 以下是模拟数据
|
||||||
|
const siteRes = [
|
||||||
|
{ id: 1, siteName: 'site1.com' },
|
||||||
|
{ id: 2, siteName: 'site2.com' },
|
||||||
|
{ id: 3, siteName: 'site2.com' },
|
||||||
|
];
|
||||||
|
// 转换为前端所需要的格式
|
||||||
|
const options = siteRes.map((item: any) => {
|
||||||
|
return {
|
||||||
|
value: item.siteName,
|
||||||
|
label: item.siteName,
|
||||||
|
domain: item.siteName,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
// 将站点域名名称根据证书域名进行匹配分组,分成匹配的和不匹配的两组选项,返回给前端,供用户选择
|
||||||
|
return optionsUtils.buildGroupOptions(options, this.certDomains);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 注意事项
|
||||||
|
|
||||||
|
1. **插件命名**:插件名称应遵循命名规范,大写字母开头,驼峰命名。
|
||||||
|
2. **类名规范**:类名应与插件名称(name)一致。
|
||||||
|
3. **证书选择**:必须包含证书选择参数,用于获取前置任务输出的域名证书。
|
||||||
|
4. **日志输出**:使用 `this.logger` 输出日志,而不是 `console`。
|
||||||
|
5. **错误处理**:执行过程中的错误应被捕获并记录。
|
||||||
|
6. **授权获取**:使用 `this.getAccess(accessId)` 获取授权信息。
|
||||||
|
|
||||||
|
## 完整示例
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
import { AbstractTaskPlugin, IsTaskPlugin, PageSearch, pluginGroups, RunStrategy, TaskInput } from '@certd/pipeline';
|
||||||
|
import { CertInfo, CertReader } from '@certd/plugin-cert';
|
||||||
|
import { createCertDomainGetterInputDefine, createRemoteSelectInputDefine } from '@certd/plugin-lib';
|
||||||
|
import { optionsUtils } from '@certd/basic';
|
||||||
|
import { CertApplyPluginNames} from '@certd/plugin-cert';
|
||||||
|
@IsTaskPlugin({
|
||||||
|
//命名规范,插件类型+功能(就是目录plugin-demo中的demo),大写字母开头,驼峰命名
|
||||||
|
name: 'DemoTest',
|
||||||
|
title: 'Demo-测试插件',
|
||||||
|
icon: 'clarity:plugin-line',
|
||||||
|
//插件分组
|
||||||
|
group: pluginGroups.other.key,
|
||||||
|
default: {
|
||||||
|
//默认值配置照抄即可
|
||||||
|
strategy: {
|
||||||
|
runStrategy: RunStrategy.SkipWhenSucceed,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
//类名规范,跟上面插件名称(name)一致
|
||||||
|
export class DemoTest extends AbstractTaskPlugin {
|
||||||
|
//测试参数
|
||||||
|
@TaskInput({
|
||||||
|
title: '属性示例',
|
||||||
|
value: '默认值',
|
||||||
|
component: {
|
||||||
|
//前端组件配置,具体配置见组件文档 https://www.antdv.com/components/input-cn
|
||||||
|
name: 'a-input',
|
||||||
|
vModel: 'value', //双向绑定组件的props名称
|
||||||
|
},
|
||||||
|
helper: '帮助说明,[链接](https://certd.docmirror.cn)',
|
||||||
|
required: false, //是否必填
|
||||||
|
})
|
||||||
|
text!: string;
|
||||||
|
|
||||||
|
//测试参数
|
||||||
|
@TaskInput({
|
||||||
|
title: '选择框',
|
||||||
|
component: {
|
||||||
|
//前端组件配置,具体配置见组件文档 https://www.antdv.com/components/select-cn
|
||||||
|
name: 'a-auto-complete',
|
||||||
|
vModel: 'value',
|
||||||
|
options: [
|
||||||
|
//选项列表
|
||||||
|
{ label: '动态显', value: 'show' },
|
||||||
|
{ label: '动态隐', value: 'hide' },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
select!: string;
|
||||||
|
|
||||||
|
@TaskInput({
|
||||||
|
title: '动态显隐',
|
||||||
|
helper: '我会根据选择框的值进行显隐',
|
||||||
|
show: true, //动态计算的值会覆盖它
|
||||||
|
//动态计算脚本, mergeScript返回的对象会合并当前配置,此处演示 show的值会被动态计算结果覆盖,show的值根据用户选择的select的值决定
|
||||||
|
mergeScript: `
|
||||||
|
return {
|
||||||
|
show: ctx.compute(({form})=>{
|
||||||
|
return form.select === 'show';
|
||||||
|
})
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
})
|
||||||
|
showText!: string;
|
||||||
|
|
||||||
|
//测试参数
|
||||||
|
@TaskInput({
|
||||||
|
title: '多选框',
|
||||||
|
component: {
|
||||||
|
//前端组件配置,具体配置见组件文档 https://www.antdv.com/components/select-cn
|
||||||
|
name: 'a-select',
|
||||||
|
vModel: 'value',
|
||||||
|
mode: 'tags',
|
||||||
|
multiple: true,
|
||||||
|
options: [
|
||||||
|
{ value: '1', label: '选项1' },
|
||||||
|
{ value: '2', label: '选项2' },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
multiSelect!: string;
|
||||||
|
|
||||||
|
//测试参数
|
||||||
|
@TaskInput({
|
||||||
|
title: 'switch',
|
||||||
|
component: {
|
||||||
|
//前端组件配置,具体配置见组件文档 https://www.antdv.com/components/switch-cn
|
||||||
|
name: 'a-switch',
|
||||||
|
vModel: 'checked',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
switch!: boolean;
|
||||||
|
//证书选择,此项必须要有
|
||||||
|
@TaskInput({
|
||||||
|
title: '域名证书',
|
||||||
|
helper: '请选择前置任务输出的域名证书',
|
||||||
|
component: {
|
||||||
|
name: 'output-selector',
|
||||||
|
from: [...CertApplyPluginNames],
|
||||||
|
},
|
||||||
|
// required: true, // 必填
|
||||||
|
})
|
||||||
|
cert!: CertInfo;
|
||||||
|
|
||||||
|
@TaskInput(createCertDomainGetterInputDefine({ props: { required: false } }))
|
||||||
|
//前端可以展示,当前申请的证书域名列表
|
||||||
|
certDomains!: string[];
|
||||||
|
|
||||||
|
//授权选择框
|
||||||
|
@TaskInput({
|
||||||
|
title: 'demo授权',
|
||||||
|
helper: 'demoAccess授权',
|
||||||
|
component: {
|
||||||
|
name: 'access-selector',
|
||||||
|
type: 'demo', //固定授权类型
|
||||||
|
},
|
||||||
|
// rules: [{ required: true, message: '此项必填' }],
|
||||||
|
// required: true, //必填
|
||||||
|
})
|
||||||
|
accessId!: string;
|
||||||
|
|
||||||
|
@TaskInput(
|
||||||
|
createRemoteSelectInputDefine({
|
||||||
|
title: '从后端获取选项',
|
||||||
|
helper: '选择时可以从后端获取选项',
|
||||||
|
action: DemoTest.prototype.onGetSiteList.name,
|
||||||
|
//当以下参数变化时,触发获取选项
|
||||||
|
watches: ['certDomains', 'accessId'],
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
siteName!: string | string[];
|
||||||
|
|
||||||
|
//插件实例化时执行的方法
|
||||||
|
async onInstance() {}
|
||||||
|
|
||||||
|
//插件执行方法
|
||||||
|
async execute(): Promise<void> {
|
||||||
|
const { select, text, cert, accessId } = this;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const access = await this.getAccess(accessId);
|
||||||
|
this.logger.debug('access', access);
|
||||||
|
} catch (e) {
|
||||||
|
this.logger.error('获取授权失败', e);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const certReader = new CertReader(cert);
|
||||||
|
this.logger.debug('certReader', certReader);
|
||||||
|
} catch (e) {
|
||||||
|
this.logger.error('读取crt失败', e);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.info('DemoTestPlugin execute');
|
||||||
|
this.logger.info('text:', text);
|
||||||
|
this.logger.info('select:', select);
|
||||||
|
this.logger.info('switch:', this.switch);
|
||||||
|
this.logger.info('授权id:', accessId);
|
||||||
|
|
||||||
|
// const res = await this.http.request({
|
||||||
|
// url: 'https://api.demo.com',
|
||||||
|
// method: 'GET',
|
||||||
|
// });
|
||||||
|
// if (res.code !== 0) {
|
||||||
|
// //检查res是否报错,你需要抛异常,来结束插件执行,否则会判定为执行成功,下次执行时会跳过本任务
|
||||||
|
// throw new Error(res.message);
|
||||||
|
// }
|
||||||
|
// this.logger.info('部署成功:', res);
|
||||||
|
}
|
||||||
|
|
||||||
|
//此方法演示,如何让前端在添加插件时可以从后端获取选项,这里是后端返回选项的方法
|
||||||
|
async onGetSiteList(req: PageSearch) {
|
||||||
|
if (!this.accessId) {
|
||||||
|
throw new Error('请选择Access授权');
|
||||||
|
}
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const access = await this.getAccess(this.accessId);
|
||||||
|
|
||||||
|
// const siteRes = await access.GetDomainList(req);
|
||||||
|
//以下是模拟数据
|
||||||
|
const siteRes = [
|
||||||
|
{ id: 1, siteName: 'site1.com' },
|
||||||
|
{ id: 2, siteName: 'site2.com' },
|
||||||
|
{ id: 3, siteName: 'site2.com' },
|
||||||
|
];
|
||||||
|
//转换为前端所需要的格式
|
||||||
|
const options = siteRes.map((item: any) => {
|
||||||
|
return {
|
||||||
|
value: item.siteName,
|
||||||
|
label: item.siteName,
|
||||||
|
domain: item.siteName,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
//将站点域名名称根据证书域名进行匹配分组,分成匹配的和不匹配的两组选项,返回给前端,供用户选择
|
||||||
|
return optionsUtils.buildGroupOptions(options, this.certDomains);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
我需要开发一个 Task 插件,用于将申请的证书部署到指定的应用系统中。请指导我如何实现。
|
||||||
@@ -0,0 +1,129 @@
|
|||||||
|
# Task 插件开发指南
|
||||||
|
|
||||||
|
## 开发步骤
|
||||||
|
|
||||||
|
### 1. 导入必要的依赖
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
import { AbstractTaskPlugin, IsTaskPlugin, PageSearch, pluginGroups, RunStrategy, TaskInput } from '@certd/pipeline';
|
||||||
|
import { CertInfo, CertReader } from '@certd/plugin-cert';
|
||||||
|
import { createCertDomainGetterInputDefine, createRemoteSelectInputDefine } from '@certd/plugin-lib';
|
||||||
|
import { optionsUtils } from '@certd/basic';
|
||||||
|
import { CertApplyPluginNames} from '@certd/plugin-cert';
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. 使用 @IsTaskPlugin 注解注册插件
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
@IsTaskPlugin({
|
||||||
|
// 命名规范,插件类型+功能,大写字母开头,驼峰命名
|
||||||
|
name: 'DemoTest',
|
||||||
|
title: 'Demo-测试插件', // 插件标题
|
||||||
|
icon: 'clarity:plugin-line', // 插件图标
|
||||||
|
// 插件分组
|
||||||
|
group: pluginGroups.other.key,
|
||||||
|
default: {
|
||||||
|
// 默认值配置照抄即可
|
||||||
|
strategy: {
|
||||||
|
runStrategy: RunStrategy.SkipWhenSucceed,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
// 类名规范,跟上面插件名称(name)一致
|
||||||
|
export class DemoTest extends AbstractTaskPlugin {
|
||||||
|
// 插件实现...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. 定义任务输入参数
|
||||||
|
|
||||||
|
使用 `@TaskInput` 注解定义任务输入参数:
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
// 测试参数
|
||||||
|
@TaskInput({
|
||||||
|
title: '属性示例',
|
||||||
|
value: '默认值',
|
||||||
|
component: {
|
||||||
|
//前端组件配置,具体配置见组件文档 https://www.antdv.com/components/input-cn
|
||||||
|
name: 'a-input',
|
||||||
|
vModel: 'value', //双向绑定组件的props名称
|
||||||
|
},
|
||||||
|
helper: '帮助说明,[链接](https://certd.docmirror.cn)',
|
||||||
|
required: false, //是否必填
|
||||||
|
})
|
||||||
|
text!: string;
|
||||||
|
|
||||||
|
//证书选择,此项必须要有
|
||||||
|
@TaskInput({
|
||||||
|
title: '域名证书',
|
||||||
|
helper: '请选择前置任务输出的域名证书',
|
||||||
|
component: {
|
||||||
|
name: 'output-selector',
|
||||||
|
from: [...CertApplyPluginNames],
|
||||||
|
},
|
||||||
|
// required: true, // 必填
|
||||||
|
})
|
||||||
|
cert!: CertInfo;
|
||||||
|
|
||||||
|
@TaskInput(createCertDomainGetterInputDefine({ props: { required: false } }))
|
||||||
|
//前端可以展示,当前申请的证书域名列表
|
||||||
|
certDomains!: string[];
|
||||||
|
|
||||||
|
//授权选择框
|
||||||
|
@TaskInput({
|
||||||
|
title: 'demo授权',
|
||||||
|
helper: 'demoAccess授权',
|
||||||
|
component: {
|
||||||
|
name: 'access-selector',
|
||||||
|
type: 'demo', //固定授权类型
|
||||||
|
},
|
||||||
|
// rules: [{ required: true, message: '此项必填' }],
|
||||||
|
// required: true, //必填
|
||||||
|
})
|
||||||
|
accessId!: string;
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. 实现插件方法
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
//插件实例化时执行的方法
|
||||||
|
async onInstance() {}
|
||||||
|
|
||||||
|
//插件执行方法
|
||||||
|
async execute(): Promise<void> {
|
||||||
|
const { select, text, cert, accessId } = this;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const access = await this.getAccess(accessId);
|
||||||
|
this.logger.debug('access', access);
|
||||||
|
} catch (e) {
|
||||||
|
this.logger.error('获取授权失败', e);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const certReader = new CertReader(cert);
|
||||||
|
this.logger.debug('certReader', certReader);
|
||||||
|
} catch (e) {
|
||||||
|
this.logger.error('读取crt失败', e);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.info('DemoTestPlugin execute');
|
||||||
|
this.logger.info('text:', text);
|
||||||
|
this.logger.info('select:', select);
|
||||||
|
this.logger.info('switch:', this.switch);
|
||||||
|
this.logger.info('授权id:', accessId);
|
||||||
|
|
||||||
|
// 具体的部署逻辑
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 注意事项
|
||||||
|
|
||||||
|
1. **插件命名**:插件名称应遵循命名规范,大写字母开头,驼峰命名。
|
||||||
|
2. **类名规范**:类名应与插件名称(name)一致。
|
||||||
|
3. **证书选择**:必须包含证书选择参数,用于获取前置任务输出的域名证书。
|
||||||
|
4. **日志输出**:使用 `this.logger` 输出日志,而不是 `console`。
|
||||||
|
5. **错误处理**:执行过程中的错误应被捕获并记录。
|
||||||
|
6. **授权获取**:使用 `this.getAccess(accessId)` 获取授权信息。
|
||||||
@@ -0,0 +1,83 @@
|
|||||||
|
{
|
||||||
|
// 使用 IntelliSense 了解相关属性。
|
||||||
|
// 悬停以查看现有属性的描述。
|
||||||
|
// 欲了解更多信息,请访问: https://go.microsoft.com/fwlink/?linkid=830387
|
||||||
|
"version": "0.2.0",
|
||||||
|
"configurations": [
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "client",
|
||||||
|
"type": "node",
|
||||||
|
"request": "launch",
|
||||||
|
"cwd": "${workspaceFolder}/packages/ui/certd-client",
|
||||||
|
"runtimeExecutable": "pnpm",
|
||||||
|
"runtimeArgs": ["dev"],
|
||||||
|
"console": "integratedTerminal",
|
||||||
|
"internalConsoleOptions": "neverOpen"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "server",
|
||||||
|
"type": "node",
|
||||||
|
"request": "launch",
|
||||||
|
"cwd": "${workspaceFolder}/packages/ui/certd-server",
|
||||||
|
"runtimeExecutable": "pnpm",
|
||||||
|
"runtimeArgs": ["dev"],
|
||||||
|
"console": "integratedTerminal",
|
||||||
|
"internalConsoleOptions": "neverOpen"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "server-mysql",
|
||||||
|
"type": "node",
|
||||||
|
"request": "launch",
|
||||||
|
"cwd": "${workspaceFolder}/packages/ui/certd-server",
|
||||||
|
"runtimeExecutable": "pnpm",
|
||||||
|
"runtimeArgs": ["dev-mysql"],
|
||||||
|
"console": "integratedTerminal",
|
||||||
|
"internalConsoleOptions": "neverOpen"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "server-pg",
|
||||||
|
"type": "node",
|
||||||
|
"request": "launch",
|
||||||
|
"cwd": "${workspaceFolder}/packages/ui/certd-server",
|
||||||
|
"runtimeExecutable": "pnpm",
|
||||||
|
"runtimeArgs": ["dev-pg"],
|
||||||
|
"console": "integratedTerminal",
|
||||||
|
"internalConsoleOptions": "neverOpen"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "server-pgpl",
|
||||||
|
"type": "node",
|
||||||
|
"request": "launch",
|
||||||
|
"cwd": "${workspaceFolder}/packages/ui/certd-server",
|
||||||
|
"runtimeExecutable": "pnpm",
|
||||||
|
"runtimeArgs": ["dev-pgpl"],
|
||||||
|
"console": "integratedTerminal",
|
||||||
|
"internalConsoleOptions": "neverOpen"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "server-common",
|
||||||
|
"type": "node",
|
||||||
|
"request": "launch",
|
||||||
|
"cwd": "${workspaceFolder}/packages/ui/certd-server",
|
||||||
|
"runtimeExecutable": "pnpm",
|
||||||
|
"runtimeArgs": ["dev-commpro"],
|
||||||
|
"console": "integratedTerminal",
|
||||||
|
"internalConsoleOptions": "neverOpen"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "server-local-plus",
|
||||||
|
"type": "node",
|
||||||
|
"request": "launch",
|
||||||
|
"cwd": "${workspaceFolder}/packages/ui/certd-server",
|
||||||
|
"runtimeExecutable": "npm",
|
||||||
|
"runtimeArgs": ["run", "dev-localplus"],
|
||||||
|
"console": "integratedTerminal",
|
||||||
|
"internalConsoleOptions": "neverOpen",
|
||||||
|
"env": {
|
||||||
|
"plus_use_prod": "false",
|
||||||
|
"PLUS_SERVER_BASE_URL": "http://127.0.0.1:11007"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
{
|
||||||
|
"eslint.debug": false,
|
||||||
|
"eslint.format.enable": true,
|
||||||
|
"typescript.tsc.autoDetect": "watch",
|
||||||
|
"git.scanRepositories": [
|
||||||
|
"./packages/pro"
|
||||||
|
],
|
||||||
|
"editor.defaultFormatter": "dbaeumer.vscode-eslint",
|
||||||
|
"[typescript]": {
|
||||||
|
"editor.defaultFormatter": "vscode.typescript-language-features"
|
||||||
|
},
|
||||||
|
"editor.tabSize": 2,
|
||||||
|
"explorer.autoReveal": false,
|
||||||
|
"[javascript]": {
|
||||||
|
"editor.defaultFormatter": "vscode.typescript-language-features"
|
||||||
|
},
|
||||||
|
"[less]": {
|
||||||
|
"editor.defaultFormatter": "vscode.css-language-features"
|
||||||
|
},
|
||||||
|
"scm.repositories.visible": 9,
|
||||||
|
"scm.repositories.explorer": false,
|
||||||
|
"scm.repositories.selectionMode": "multiple",
|
||||||
|
"scm.repositories.sortOrder": "discovery time"
|
||||||
|
}
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
{
|
||||||
|
"version": "2.0.0",
|
||||||
|
"tasks": [
|
||||||
|
{
|
||||||
|
"label": "启动Client",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "npm",
|
||||||
|
"args": ["run", "dev"],
|
||||||
|
"options": {
|
||||||
|
"cwd": "${workspaceFolder}/packages/ui/certd-client"
|
||||||
|
},
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": false,
|
||||||
|
"panel": "shared"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "启动Server",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "npm",
|
||||||
|
"args": ["run", "dev"],
|
||||||
|
"options": {
|
||||||
|
"cwd": "${workspaceFolder}/packages/ui/certd-server"
|
||||||
|
},
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": false,
|
||||||
|
"panel": "shared"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "同时启动Client和Server",
|
||||||
|
"dependsOn": ["启动Client", "启动Server"],
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher": []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1,30 +1,48 @@
|
|||||||
# Certd
|
# Certd
|
||||||
|
|
||||||
Certd 是一个免费全自动申请和自动部署更新SSL证书的管理系统。
|
中文 | [English](./README_en.md)
|
||||||
后缀d取自linux守护进程的命名风格,意为证书守护进程。
|
|
||||||
|
Certd® 是一个免费的全自动证书管理系统,让你的网站证书永不过期。
|
||||||
|
后缀d取自linux守护进程的命名风格,意为证书守护进程
|
||||||
|
|
||||||
|
|
||||||
|
>首创流水线申请部署证书模式,已被多个项目“借鉴”,被抄也是一种成功。
|
||||||
|
|
||||||
|
> 关于证书续期:
|
||||||
|
>* 实际上没有办法不改变证书文件本身情况下直接续期或者续签。
|
||||||
|
>* 我们所说的续期,其实就是按照全套流程重新申请一份新证书,然后重新部署上去。
|
||||||
|
>* 免费证书过期时间90天,以后可能还会缩短,所以自动化部署必不可少
|
||||||
|
|
||||||
|
|
||||||
|
> 流水线数量现已调整为无限制,欢迎大家使用
|
||||||
|
|
||||||
|
|
||||||
|
|官方开源地址: | |
|
||||||
|
| ---- | ---- |
|
||||||
|
| [Github](https://github.com/certd/certd)|  |
|
||||||
|
| [Gitee](https://gitee.com/certd/certd) |  |
|
||||||
|
| [AtomGit](https://atomgit.com/certd/certd) | |
|
||||||
|
|
||||||
关键字:证书自动申请、证书自动更新、证书自动续期、证书自动续签、证书管理工具
|
|
||||||
|
|
||||||
## 一、特性
|
## 一、特性
|
||||||
本项目不仅支持证书申请过程自动化,还可以自动化部署更新证书,让你的证书永不过期。
|
本项目不仅支持证书申请过程自动化,还可以自动化部署更新证书,让你的证书永不过期。
|
||||||
|
|
||||||
* 全自动申请证书(支持所有注册商注册的域名)
|
* **全自动申请证书**: 支持所有注册商注册的域名,支持DNS-01、HTTP-01、CNAME代理等多种域名验证方式
|
||||||
* 全自动部署更新证书(目前支持部署到主机、阿里云、腾讯云等,目前已支持60+部署插件)
|
* **全自动部署更新证书**: 目前支持部署到主机、阿里云、腾讯云等110+部署插件
|
||||||
* 支持DNS-01、HTTP-01、CNAME代理等多种域名验证方式
|
* **多种证书格式**: 支持pem、pfx、der、jks、p7b
|
||||||
* 支持通配符域名/泛域名,支持多个域名打到一个证书上,支持pem、pfx、der、jks等多种证书格式
|
* **免费通配符域名/泛域名证书**: 支持多个域名打到一个证书上
|
||||||
* 邮件通知、webhook通知
|
* **多种通知方式**: 邮件通知、webhook通知、企微、钉钉、飞书、anpush等多种通知方式
|
||||||
* 私有化部署,数据保存本地,授权信息加密存储,镜像由Github Actions构建,过程公开透明
|
* **私有化部署**: 数据保存本地,安装简单快捷,镜像由Github Actions构建,过程公开透明
|
||||||
* 支持SQLite,PostgreSQL、MySQL数据库
|
* **多重安全保障**: 授权加密,站点隐藏,2FA,密码防爆破等多重安全保障
|
||||||
|
* **多数据库支持**:支持SQLite、PostgreSQL、MySQL
|
||||||
|
* **开放接口支持**: 提供RESTful API接口,方便集成到其他系统
|
||||||
|
* **站点证书监控**: 定时监控网站证书的过期时间
|
||||||
|
* **多用户管理**: 用户可以管理自己的证书流水线
|
||||||
|
* **多语言支持**: 中英双语切换
|
||||||
|
* **无忧升级**: 版本向下兼容
|
||||||
|
|
||||||
>
|

|
||||||
> 流水线数量现已调整为无限制,欢迎大家使用
|
|
||||||
>
|
|
||||||
|
|
||||||
> 关于证书续期:
|
|
||||||
>* 实际上没有办法不改变证书文件本身情况下直接续期或者续签。
|
|
||||||
>* 我们所说的续期,其实就是按照全套流程重新申请一份新证书,然后重新部署上去。
|
|
||||||
|
|
||||||
|
|
||||||
## 二、在线体验
|
## 二、在线体验
|
||||||
|
|
||||||
官方Demo地址,自助注册后体验
|
官方Demo地址,自助注册后体验
|
||||||
@@ -62,7 +80,7 @@ https://certd.handfree.work/
|
|||||||
-------> [点我查看详细使用步骤演示](./step.md) <--------
|
-------> [点我查看详细使用步骤演示](./step.md) <--------
|
||||||
↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑
|
↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑
|
||||||
|
|
||||||
更多教程请访问文档网站 [certd.docmirror.cn](https://certd.docmirror.cn/)
|
更多教程请访问官方文档 [certd.docmirror.cn](https://certd.docmirror.cn/guide/)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -72,10 +90,12 @@ https://certd.handfree.work/
|
|||||||
|
|
||||||
您可以根据实际情况从如下方式中选择一种方式进行私有化部署:
|
您可以根据实际情况从如下方式中选择一种方式进行私有化部署:
|
||||||
|
|
||||||
1. [宝塔面板方式部署](https://certd.docmirror.cn/guide/install/docker/)
|
1. 【推荐】[Docker方式部署 ](https://certd.docmirror.cn/guide/install/docker/)
|
||||||
2. [1Panel面板方式部署](https://certd.docmirror.cn/guide/install/1panel/)
|
2. 【推荐】[宝塔面板方式部署 ](https://certd.docmirror.cn/guide/install/docker/)
|
||||||
3. [Docker方式部署](https://certd.docmirror.cn/guide/install/docker/)
|
3. 【推荐】[1Panel面板方式部署](https://certd.docmirror.cn/guide/install/1panel/)
|
||||||
4. [源码方式部署](https://certd.docmirror.cn/guide/install/source/)
|
4. 【推荐】[雨云一键部署](https://app.rainyun.com/apps/rca/store/6646/?ref=NzExMDQ2) : 首充翻倍,每月仅需2.2元
|
||||||
|
[<img src="https://rainyun-apps.cn-nb1.rains3.com/materials/deploy-on-rainyun-cn.svg">](https://app.rainyun.com/apps/rca/store/6646/?ref=NzExMDQ2)
|
||||||
|
5. 【不推荐】[源码方式部署 ](https://certd.docmirror.cn/guide/install/source/)
|
||||||
|
|
||||||
#### Docker镜像说明:
|
#### Docker镜像说明:
|
||||||
* 国内镜像地址:
|
* 国内镜像地址:
|
||||||
@@ -100,87 +120,30 @@ https://certd.handfree.work/
|
|||||||
> * 请务必使用web应用防火墙防护本应用,防止XSS、SQL注入等攻击
|
> * 请务必使用web应用防火墙防护本应用,防止XSS、SQL注入等攻击
|
||||||
> * 请务必做好服务器本身的安全防护,防止数据库泄露
|
> * 请务必做好服务器本身的安全防护,防止数据库泄露
|
||||||
> * 请务必做好数据备份,避免数据丢失
|
> * 请务必做好数据备份,避免数据丢失
|
||||||
|
> * [更多安全生产建议点我](https://certd.docmirror.cn/guide/feature/safe/)
|
||||||
## 五、 升级
|
|
||||||
|
|
||||||
### docker-compose方式部署
|
|
||||||
#### 1. 如果使用固定版本号
|
|
||||||
1. 修改`docker-compose.yaml`中的镜像版本号
|
|
||||||
2. 运行`docker compose up -d` 即可
|
|
||||||
|
|
||||||
#### 2. 如果需要使用最新版本
|
|
||||||
```shell
|
|
||||||
#重新拉取镜像
|
|
||||||
docker pull registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest
|
|
||||||
# 重新启动容器
|
|
||||||
docker compose down
|
|
||||||
docker compose up -d
|
|
||||||
```
|
|
||||||
> 数据默认存在`/data/certd`目录下,不用担心数据丢失
|
|
||||||
|
|
||||||
### 自动升级(仅限尝鲜建议非生产使用)
|
|
||||||
```yaml
|
|
||||||
version: '3.3'
|
|
||||||
services:
|
|
||||||
certd:
|
|
||||||
image: registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest
|
|
||||||
container_name: certd
|
|
||||||
restart: unless-stopped
|
|
||||||
volumes:
|
|
||||||
- /data/certd:/app/data
|
|
||||||
ports:
|
|
||||||
- "7001:7001"
|
|
||||||
- "7002:7002"
|
|
||||||
environment:
|
|
||||||
- certd_system_resetAdminPasswd=false
|
|
||||||
labels:
|
|
||||||
com.centurylinklabs.watchtower.enable: "true"
|
|
||||||
|
|
||||||
certd-updater: # 添加 Watchtower 服务
|
|
||||||
image: containrrr/watchtower:latest
|
|
||||||
container_name: certd-updater
|
|
||||||
restart: unless-stopped
|
|
||||||
volumes:
|
|
||||||
- /var/run/docker.sock:/var/run/docker.sock
|
|
||||||
# 配置 自动更新
|
|
||||||
environment:
|
|
||||||
- WATCHTOWER_CLEANUP=true # 自动清理旧版本容器
|
|
||||||
- WATCHTOWER_INCLUDE_STOPPED=false # 不更新已停止的容器
|
|
||||||
- WATCHTOWER_LABEL_ENABLE=true # 根据容器标签进行更新
|
|
||||||
- WATCHTOWER_POLL_INTERVAL=300 # 每 5 分钟检查一次更新
|
|
||||||
|
|
||||||
```
|
|
||||||
|
|
||||||
### 其他部署方式升级方法
|
|
||||||
请参考 https://certd.docmirror.cn/guide/install/upgrade.html
|
|
||||||
|
|
||||||
|
|
||||||
|
## 五、生态
|
||||||
|
|
||||||
|
### 1. 客户端工具 SSL-Assistant
|
||||||
|
`SSL Assistant` 是一个运行于主机上的证书部署管理助手客户端。
|
||||||
|
支持自动扫描主机`Nginx`配置,然后从`Certd`拉取证书并部署。
|
||||||
|
在不想暴露ssh主机密码情况下,该工具非常好用。
|
||||||
|
|
||||||
### 更新日志:
|
开源地址: https://github.com/Youngxj/SSL-Assistant
|
||||||
|
|
||||||
[CHANGELOG](./CHANGELOG.md)
|
|
||||||
|
|
||||||
|
|
||||||
## 六、一些说明
|
## 六、更多帮助
|
||||||
* 本项目ssl证书提供商为letencrypt/Google/ZeroSSL
|
请访问官方文档:[https://certd.docmirror.cn/](https://certd.docmirror.cn/guide/)
|
||||||
* 申请过程遵循acme协议
|
|
||||||
* 证书续期:
|
* 升级方法:[升级方法](https://certd.docmirror.cn/guide/install/upgrade/)
|
||||||
* 实际上没有办法不改变证书文件本身情况下直接续期或者续签。
|
* 常见问题:[忘记密码](https://certd.docmirror.cn/guide/use/forgotpasswd/)
|
||||||
* 我们所说的续期,其实就是按照全套流程重新申请一份新证书,然后重新部署上去。
|
* 多数据库:[多数据库配置](https://certd.docmirror.cn/guide/install/database/)
|
||||||
* 免费证书过期时间90天,以后可能还会缩短,所以自动化部署必不可少
|
* 站点安全:[站点安全特性](https://certd.docmirror.cn/guide/feature/safe/)
|
||||||
* 设置每天自动运行,当证书过期前35天,会自动重新申请证书并部署
|
* 更新日志:[CHANGELOG](./CHANGELOG.md)
|
||||||
|
|
||||||
|
|
||||||
## 七、不同平台的设置说明
|
## 七、联系作者
|
||||||
|
|
||||||
* 已迁移到新的文档网站,请到常见问题章节查看
|
|
||||||
* [最新文档站链接 https://certd.docmirror.cn](https://certd.docmirror.cn/)
|
|
||||||
|
|
||||||
## 八、问题处理
|
|
||||||
### 7.1 忘记管理员密码
|
|
||||||
[重置管理员密码方法](https://certd.docmirror.cn/guide/use/forgotpasswd/)
|
|
||||||
|
|
||||||
## 九、联系作者
|
|
||||||
如有疑问,欢迎加入群聊(请备注certd)
|
如有疑问,欢迎加入群聊(请备注certd)
|
||||||
|
|
||||||
| 加群 | 微信群 | QQ群 |
|
| 加群 | 微信群 | QQ群 |
|
||||||
@@ -194,58 +157,57 @@ services:
|
|||||||
| 二维码 | <img height="230" src="./docs/guide/contact/images/me.png"> |
|
| 二维码 | <img height="230" src="./docs/guide/contact/images/me.png"> |
|
||||||
|
|
||||||
|
|
||||||
## 十、捐赠
|
## 八、赞助捐赠
|
||||||
************************
|
|
||||||
支持开源,为爱发电,我已入驻爱发电
|
|
||||||
https://afdian.com/a/greper
|
|
||||||
|
|
||||||
发电权益:
|
开源为什么要做专业版收费?
|
||||||
1. 可加入发电专属群,可以获得作者一对一技术支持
|
1. 纯靠为爱发电不可持续(比如:我的[dev-sidecar项目](https://github.com/docmirror/dev-sidecar)即便是拥有20K+star,也差点凉凉,幸亏有另外大佬接手用爱发电)
|
||||||
2. 您的需求我们将优先实现,并且将作为专业版功能提供
|
2. 没有赞助的项目,作者会比较任性,不会用心倾听用户的心声,不顾用户体验(比如:下意识拒绝需求、频繁破坏性变更升级、全盘推倒重来之类的)
|
||||||
3. 一年期专业版激活码
|
3. 没有赞助的项目,交流群的戾气有时候比较重,容易起冲突
|
||||||
|
|
||||||
|
赞助权益:
|
||||||
|
1. 可加入专属VIP群,可以获得作者一对一技术支持,必要时可以远程协助
|
||||||
|
2. 您的需求我们将优先实现,并且可能将作为专业版功能提供
|
||||||
|
3. 获得专业版功能
|
||||||
|
|
||||||
|
[50元专业版优惠券限时领取](https://app.handfree.work/subject/#/app/certd/product)
|
||||||
|
|
||||||
专业版特权对比
|
专业版特权对比
|
||||||
|
|
||||||
| 功能 | 免费版 | 专业版 |
|
| 功能 | 免费版 | 专业版 |
|
||||||
|---------|--------------------|-----------------------------|
|
|---------|---------------------------------------|--------------------------------|
|
||||||
| 免费证书申请 | 免费无限制 | 免费无限制 |
|
| 免费证书申请 | 免费无限制 | 免费无限制 |
|
||||||
| 域名数量 | 无限制 | 无限制 |
|
| 证书域名数量 | 无限制 | 无限制 |
|
||||||
| 证书流水线条数 | 无限制 | 无限制 |
|
| 证书流水线条数 | 无限制 | 无限制 |
|
||||||
| 站点证书监控 | 限制1条 | 无限制 |
|
| 自动部署插件 | 阿里云CDN、腾讯云、七牛CDN、主机部署、宝塔、1Panel等大部分插件 | 群晖、威联通、proxmox等 |
|
||||||
| 自动部署插件 | 阿里云CDN、腾讯云、七牛CDN、主机部署等 | 支持群晖、宝塔、1Panel等,持续开发中 |
|
| 通知 | 邮件通知、自定义webhook | 邮件免配置、企微、钉钉、飞书、anpush、server酱等 |
|
||||||
| 通知 | 邮件通知、自定义webhook | 邮件免配置、企微、飞书、anpush、server酱等 |
|
| 站点监控 | 限制1条 | 无限制 |
|
||||||
|
| 批量操作 | 无 | 流水线模版,流水线复制,批量运行,批量设置通知、定时等 |
|
||||||
|
| VIP群 | 无 | 可加,一对一技术支持,必要时可申请远程协助 |
|
||||||
|
|
||||||
|
|
||||||
************************
|
## 九、贡献代码
|
||||||
|
|
||||||
## 十一、贡献代码
|
1. 本地开发请参考 [贡献插件向导](https://certd.docmirror.cn/guide/development/)
|
||||||
|
|
||||||
1. 本地开发 [贡献插件](https://certd.docmirror.cn/guide/development/)
|
|
||||||
2. 作为贡献者,代表您同意您贡献的代码如下许可:
|
2. 作为贡献者,代表您同意您贡献的代码如下许可:
|
||||||
1. 可以调整开源协议以使其更严格或更宽松。
|
1. 可以调整开源协议以使其更严格或更宽松。
|
||||||
2. 可以用于商业用途。
|
2. 可以用于商业用途。
|
||||||
|
|
||||||
|
感谢以下贡献者做出的贡献。
|
||||||
|
|
||||||
|
<a href="https://github.com/certd/certd/graphs/contributors">
|
||||||
|
<img src="https://contrib.rocks/image?repo=certd/certd" />
|
||||||
|
</a>
|
||||||
|
|
||||||
## 十二、 开源许可
|
## 十、 开源许可
|
||||||
* 本项目遵循 GNU Affero General Public License(AGPL)开源协议。
|
* 本项目遵循 GNU Affero General Public License(AGPL)开源协议。
|
||||||
* 允许个人和公司内部自由使用、复制、修改和分发本项目,未获得商业授权情况下禁止任何形式的商业用途
|
* 允许个人和公司内部自由使用、复制、修改和分发本项目,未获得商业授权情况下禁止任何形式的商业用途
|
||||||
* 未获得商业授权情况下,禁止任何对logo、版权信息及授权许可相关代码的修改。
|
* 未获得商业授权情况下,禁止任何对logo、版权信息及授权许可相关代码的修改。
|
||||||
* 如需商业授权,请联系作者。
|
* 如需商业授权,请联系作者。
|
||||||
|
|
||||||
|
|
||||||
## 十三、我的其他项目(求Star)
|
## 十一、我的其他项目(求Star)
|
||||||
|
|
||||||
| 项目名称 | stars | 项目描述 |
|
| 项目名称 | stars | 项目描述 |
|
||||||
|---------------------------------------------------------|-------------------------------------------------------------------------------------------------------|-----------------------------------|
|
| --------- |--------- |----------- |
|
||||||
| [袖手AI](https://ai.handsfree.work/) | | 袖手GPT,国内可用,无需FQ,每日免费额度 |
|
|
||||||
| [fast-crud](https://gitee.com/fast-crud/fast-crud/) | <img alt="GitHub stars" src="https://img.shields.io/github/stars/fast-crud/fast-crud?logo=github"/> | 基于vue3的crud快速开发框架 |
|
| [fast-crud](https://gitee.com/fast-crud/fast-crud/) | <img alt="GitHub stars" src="https://img.shields.io/github/stars/fast-crud/fast-crud?logo=github"/> | 基于vue3的crud快速开发框架 |
|
||||||
| [dev-sidecar](https://github.com/docmirror/dev-sidecar/) | <img alt="GitHub stars" src="https://img.shields.io/github/stars/docmirror/dev-sidecar?logo=github"/> | 直连访问github工具,无需FQ,解决github无法访问的问题 |
|
| [dev-sidecar](https://github.com/docmirror/dev-sidecar/) | <img alt="GitHub stars" src="https://img.shields.io/github/stars/docmirror/dev-sidecar?logo=github"/> | 直连访问github工具,无需FQ,解决github无法访问的问题 |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## 十四、更新日志
|
|
||||||
|
|
||||||
更新日志:[CHANGELOG](./CHANGELOG.md)
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,194 @@
|
|||||||
|
# Certd
|
||||||
|
|
||||||
|
[中文](./README.md) | English
|
||||||
|
|
||||||
|
Certd® is a free, fully automated certificate management system that ensures your website certificates never expire. The suffix 'd' is inspired by the naming convention of Linux daemons, representing a certificate daemon.
|
||||||
|
|
||||||
|
> We pioneered the pipeline-based certificate application and deployment model, which has been "referenced" by multiple projects. Being copied is also a form of success.
|
||||||
|
|
||||||
|
> Regarding certificate renewal:
|
||||||
|
>* In fact, it's impossible to renew or reissue a certificate without modifying the certificate file itself.
|
||||||
|
>* What we refer to as renewal is essentially applying for a new certificate following the full process and redeploying it.
|
||||||
|
>* Free certificates expire in 90 days, which may be shortened in the future. Therefore, automated deployment is essential.
|
||||||
|
|
||||||
|
> The number of pipelines is now unlimited. Welcome to use it.
|
||||||
|
|
||||||
|
|
||||||
|
Official Open Source Address:
|
||||||
|
|
||||||
|
[Github](https://github.com/certd/certd) 
|
||||||
|
[Gitee](https://gitee.com/certd/certd) 
|
||||||
|
[AtomGit](https://atomgit.com/certd/certd) 
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## 1. Features
|
||||||
|
This project not only supports automated certificate application but also automated certificate deployment and updates, ensuring your certificates never expire.
|
||||||
|
|
||||||
|
* Fully automated certificate application (supports domains registered with all registrars and multiple domain verification methods such as DNS-01, HTTP-01, and CNAME proxy).
|
||||||
|
* Fully automated certificate deployment and updates (currently supports deployment to over 70 plugins, including hosts, Alibaba Cloud, Tencent Cloud, etc.).
|
||||||
|
* Supports wildcard domains/pan-domains, allows multiple domains in a single certificate, and supports various certificate formats such as pem, pfx, der, and jks.
|
||||||
|
* Multiple notification methods, including email, webhook, WeChat Work, DingTalk, Lark, and anpush.
|
||||||
|
* On-premises deployment, local data storage, simple and quick installation. Images are built by Github Actions, with a transparent process.
|
||||||
|
* Multiple security measures, including authorization encryption, site hiding, 2FA, and password brute-force protection.
|
||||||
|
* Supports multiple databases such as SQLite, PostgreSQL, and MySQL.
|
||||||
|
* Open API support.
|
||||||
|
* Site certificate monitoring.
|
||||||
|
* Multi-user management.
|
||||||
|
* Multi-language support (Chinese and English switching).
|
||||||
|
* Downward compatibility across all versions, with one-click worry-free upgrades.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## 2. Online Experience
|
||||||
|
Visit the official demo site and register to experience it.
|
||||||
|
|
||||||
|
https://certd.handfree.work/
|
||||||
|
|
||||||
|
> Note: Data will be cleaned up irregularly, and scheduled tasks may be stopped. For production use, please deploy it yourself.
|
||||||
|
> The content contains sensitive information. Make sure to deploy it locally for production use.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## 3. Usage Tutorial
|
||||||
|
Just 3 steps to ensure your certificates never expire.
|
||||||
|
|
||||||
|
### 1. Create a Certificate Pipeline
|
||||||
|

|
||||||
|
|
||||||
|
> After successful addition, you can directly run the pipeline to apply for a certificate.
|
||||||
|
|
||||||
|
### 2. Add a Deployment Task
|
||||||
|
Normally, we need to deploy certificates to applications. Certd supports a wide range of deployment plugins. You can choose based on your needs, such as deploying to Nginx, Alibaba Cloud, Tencent Cloud, K8S, CDN, Baota, 1Panel, etc.
|
||||||
|
|
||||||
|
Here's a demonstration of deploying certificates to a host's Nginx:
|
||||||
|

|
||||||
|
|
||||||
|
If the current deployment plugins don't meet your needs, you can also download them manually and deploy them yourself.
|
||||||
|

|
||||||
|
|
||||||
|
### 3. Run Scheduled Tasks
|
||||||
|

|
||||||
|
|
||||||
|
↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓
|
||||||
|
-------> [Click here to view detailed usage steps](./step.md) <--------
|
||||||
|
↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑
|
||||||
|
|
||||||
|
For more tutorials, please visit the official documentation [certd.docmirror.cn](https://certd.docmirror.cn/guide/).
|
||||||
|
|
||||||
|
## 4. On-Premises Deployment
|
||||||
|
Since certificates, authorization information, and other data are highly sensitive, please make sure to deploy them on-premises to ensure data security.
|
||||||
|
|
||||||
|
You can choose one of the following deployment methods based on your needs:
|
||||||
|
|
||||||
|
1. 【Recommended】[Docker Deployment](https://certd.docmirror.cn/guide/install/docker/)
|
||||||
|
2. 【Recommended】[BT Panel Deployment](https://certd.docmirror.cn/guide/install/docker/)
|
||||||
|
3. 【Recommended】[1Panel Deployment](https://certd.docmirror.cn/guide/install/1panel/)
|
||||||
|
4. 【Recommended】[Rainyun One-Click Deployment](https://app.rainyun.com/apps/rca/store/6646/?ref=NzExMDQ2_): Double your first recharge, only $2.2 per month.
|
||||||
|
[<img src="https://rainyun-apps.cn-nb1.rains3.com/materials/deploy-on-rainyun-cn.svg">](https://app.rainyun.com/apps/rca/store/6646/?ref=NzExMDQ2_)
|
||||||
|
5. 【Not Recommended】[Source Code Deployment](https://certd.docmirror.cn/guide/install/source/)
|
||||||
|
|
||||||
|
#### Docker Image Information:
|
||||||
|
* Domestic Image Addresses:
|
||||||
|
* `registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest`
|
||||||
|
* `registry.cn-shenzhen.aliyuncs.com/handsfree/certd:armv7`, `[version]-armv7`
|
||||||
|
* DockerHub Addresses:
|
||||||
|
* `https://hub.docker.com/r/greper/certd`
|
||||||
|
* `greper/certd:latest`
|
||||||
|
* `greper/certd:armv7`, `greper/certd:[version]-armv7`
|
||||||
|
* GitHub Packages Addresses:
|
||||||
|
* `ghcr.io/certd/certd:latest`
|
||||||
|
* `ghcr.io/certd/certd:armv7`, `ghcr.io/certd/certd:[version]-armv7`
|
||||||
|
|
||||||
|
* Images are built automatically by `Actions`, with a transparent process. Please use them with confidence.
|
||||||
|
* [Click here to view image build logs](https://github.com/certd/certd/actions/workflows/build-image.yml)
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
> Note:
|
||||||
|
> * The certificates, authorization information, and other data stored in this application are highly sensitive. Please take appropriate security measures.
|
||||||
|
> * Make sure to use the HTTPS protocol to access this application to avoid man-in-the-middle attacks.
|
||||||
|
> * Make sure to use a web application firewall to protect this application from attacks such as XSS and SQL injection.
|
||||||
|
> * Make sure to secure the server itself to prevent database leakage.
|
||||||
|
> * Make sure to back up your data to avoid data loss.
|
||||||
|
> * [Click here for more production safety suggestions](https://certd.docmirror.cn/guide/feature/safe/)
|
||||||
|
|
||||||
|
## 5. Ecosystem
|
||||||
|
|
||||||
|
### 1. Client Tool: SSL-Assistant
|
||||||
|
`SSL Assistant` is a certificate deployment and management assistant client that runs on hosts. It supports automatic scanning of the host's `Nginx` configuration and pulling certificates from `Certd` for deployment. This tool is very useful when you don't want to expose your SSH host password.
|
||||||
|
|
||||||
|
Open-source Address: https://github.com/Youngxj/SSL-Assistant
|
||||||
|
|
||||||
|
## 6. More Help
|
||||||
|
Please visit the official documentation: [https://certd.docmirror.cn/](https://certd.docmirror.cn/guide/).
|
||||||
|
|
||||||
|
* Upgrade Method: [Upgrade Guide](https://certd.docmirror.cn/guide/install/upgrade/)
|
||||||
|
* Common Issues: [Forgot Password](https://certd.docmirror.cn/guide/use/forgotpasswd/)
|
||||||
|
* Multi-Database: [Multi-Database Configuration](https://certd.docmirror.cn/guide/install/database/)
|
||||||
|
* Site Security: [Site Security Features](https://certd.docmirror.cn/guide/feature/safe/)
|
||||||
|
* Changelog: [CHANGELOG](./CHANGELOG.md)
|
||||||
|
|
||||||
|
## 7. Contact the Author
|
||||||
|
If you have any questions, feel free to join the group chat (please mention 'certd' in your message).
|
||||||
|
|
||||||
|
| Join Group | WeChat Group | QQ Group |
|
||||||
|
|---------|-------|-------|
|
||||||
|
| QR Code | <img height="230" src="./docs/guide/contact/images/wx.png"> | <img height="230" src="./docs/guide/contact/images/qq.png"> |
|
||||||
|
|
||||||
|
You can also add the author as a friend.
|
||||||
|
|
||||||
|
| Add Author as Friend | WeChat QQ |
|
||||||
|
|---------|-------|-------|
|
||||||
|
| QR Code | <img height="230" src="./docs/guide/contact/images/me.png"> |
|
||||||
|
|
||||||
|
## 8. Donation
|
||||||
|
************************
|
||||||
|
[](https://github.com/sponsors/greper)
|
||||||
|
************************
|
||||||
|
Support open-source projects and contribute with love. I've joined Afdian.
|
||||||
|
https://afdian.com/a/greper
|
||||||
|
|
||||||
|
Benefits of Contribution:
|
||||||
|
1. Join the exclusive contributor group and get one-on-one technical support from the author.
|
||||||
|
2. Your requests will be prioritized and implemented as professional edition features.
|
||||||
|
3. Receive a one-year professional edition activation code.
|
||||||
|
|
||||||
|
Comparison of Professional Edition Privileges:
|
||||||
|
|
||||||
|
| Feature | Free Edition | Professional Edition |
|
||||||
|
|---------|---------------------------------------|--------------------------------|
|
||||||
|
| Free Certificate Application | Unlimited for free | Unlimited for free |
|
||||||
|
| Number of Domains | Unlimited | Unlimited |
|
||||||
|
| Number of Certificate Pipelines | Unlimited | Unlimited |
|
||||||
|
| Site Certificate Monitoring | Limited to 1 | Unlimited |
|
||||||
|
| Automatic Deployment Plugins | Most plugins such as Alibaba Cloud CDN, Tencent Cloud, QiNiu CDN, Host Deployment, Baota, 1Panel | Synology |
|
||||||
|
| Notifications | Email, Custom Webhook | Email without configuration, WeChat Work, DingTalk, Lark, anpush, ServerChan, etc. |
|
||||||
|
|
||||||
|
************************
|
||||||
|
|
||||||
|
## 9. Contribute Code
|
||||||
|
|
||||||
|
1. For local development, please refer to the [Plugin Contribution Guide](https://certd.docmirror.cn/guide/development/).
|
||||||
|
2. As a contributor, you agree that your contributed code is subject to the following license:
|
||||||
|
1. The open-source license can be adjusted to be more or less restrictive.
|
||||||
|
2. It can be used for commercial purposes.
|
||||||
|
|
||||||
|
Thank you to the following contributors.
|
||||||
|
|
||||||
|
<a href="https://github.com/certd/certd/graphs/contributors">
|
||||||
|
<img src="https://contrib.rocks/image?repo=certd/certd" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
## 10. Open-Source License
|
||||||
|
* This project follows the GNU Affero General Public License (AGPL).
|
||||||
|
* Individuals and companies are allowed to use, copy, modify, and distribute this project freely for internal use. Any form of commercial use is prohibited without obtaining commercial authorization.
|
||||||
|
* Without commercial authorization, any modification of the logo, copyright information, and license-related code is prohibited.
|
||||||
|
* For commercial authorization, please contact the author.
|
||||||
|
|
||||||
|
## 11. My Other Projects (Please Star)
|
||||||
|
|
||||||
|
| Project Name | Stars | Project Description |
|
||||||
|
|----------------|---------------|--------------|
|
||||||
|
| [fast-crud](https://gitee.com/fast-crud/fast-crud/) | <img alt="GitHub stars" src="https://img.shields.io/github/stars/fast-crud/fast-crud?logo=github"/> | A fast CRUD development framework based on Vue3. |
|
||||||
|
| [dev-sidecar](https://github.com/docmirror/dev-sidecar/) | <img alt="GitHub stars" src="https://img.shields.io/github/stars/docmirror/dev-sidecar?logo=github"/> | A tool to access GitHub directly without a VPN, solving the problem of inaccessible GitHub. |
|
||||||
@@ -1 +0,0 @@
|
|||||||
2
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
01:33
|
|
||||||
@@ -3,18 +3,23 @@ services:
|
|||||||
certd:
|
certd:
|
||||||
# 镜像 # ↓↓↓↓↓ ---- 镜像版本号,建议改成固定版本号,例如:certd:1.29.0
|
# 镜像 # ↓↓↓↓↓ ---- 镜像版本号,建议改成固定版本号,例如:certd:1.29.0
|
||||||
image: registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest
|
image: registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest
|
||||||
|
# image: ghcr.io/certd/certd:latest # --------- 如果 报镜像not found,可以尝试其他镜像源
|
||||||
|
# image: greper/certd:latest
|
||||||
container_name: certd # 容器名
|
container_name: certd # 容器名
|
||||||
restart: unless-stopped # 自动重启
|
restart: unless-stopped # 自动重启
|
||||||
volumes:
|
volumes:
|
||||||
# ↓↓↓↓↓ -------------------------------------------------------- 数据库以及证书存储路径,默认存在宿主机的/data/certd/目录下,【您需要定时备份此目录,以保障数据容灾】
|
# ↓↓↓↓↓ -------------------------------------------------------- 数据库以及证书存储路径,默认存在宿主机的/data/certd/目录下,【您需要定时备份此目录,以保障数据容灾】
|
||||||
# 只要修改冒号前面的,冒号后面的/app/data不要动
|
- /data/certd:/app/data # 只要修改冒号前面的,冒号后面的/app/data切记切记不要动
|
||||||
- /data/certd:/app/data
|
#- /volume1/docker/certd:/app/data:delegated #群晖使用这个配置
|
||||||
|
# ↓↓↓↓↓ -------------------------------------------------------- 如果走时不准,考虑挂载localtime文件
|
||||||
|
#- /etc/localtime:/etc/localtime
|
||||||
|
#- /etc/timezone:/etc/timezone
|
||||||
ports: # 端口映射
|
ports: # 端口映射
|
||||||
# ↓↓↓↓ ---------------------------------------------------------- 如果端口有冲突,可以修改第一个7001为其他不冲突的端口号,第二个7001不要动
|
# ↓↓↓↓ ---------------------------------------------------------- 如果端口有冲突,可以修改第一个7001为其他不冲突的端口号,第二个7001不要动
|
||||||
- "7001:7001"
|
- "7001:7001"
|
||||||
# ↓↓↓↓ ---------------------------------------------------------- https端口,可以根据实际情况,是否暴露该端口
|
# ↓↓↓↓ ---------------------------------------------------------- https端口,可以根据实际情况,是否暴露该端口
|
||||||
- "7002:7002"
|
- "7002:7002"
|
||||||
#↓↓↓↓ -------------------------------------------------------------- 如果出现getaddrinfo ENOTFOUND错误,可以尝试设置dns
|
#↓↓↓↓ -------------------------------------------------------------- 如果出现getaddrinfo EAI_AGAIN 或 getaddrinfo ENOTFOUND 错误,可以尝试设置dns
|
||||||
# dns:
|
# dns:
|
||||||
# - 223.5.5.5 # 阿里云公共dns
|
# - 223.5.5.5 # 阿里云公共dns
|
||||||
# - 223.6.6.6
|
# - 223.6.6.6
|
||||||
@@ -35,11 +40,15 @@ services:
|
|||||||
# networks:
|
# networks:
|
||||||
# - ip6net
|
# - ip6net
|
||||||
environment:
|
environment:
|
||||||
|
# ↓↓↓↓ ----------------------------------------------------- 使用上海东八时区
|
||||||
|
- TZ=Asia/Shanghai
|
||||||
# 设置环境变量即可自定义certd配置
|
# 设置环境变量即可自定义certd配置
|
||||||
# 配置项见: packages/ui/certd-server/src/config/config.default.ts
|
# 配置项见: packages/ui/certd-server/src/config/config.default.ts
|
||||||
# 配置规则: certd_ + 配置项, 点号用_代替
|
# 配置规则: certd_ + 配置项, 点号用_代替
|
||||||
# #↓↓↓↓ ----------------------------- 如果忘记管理员密码,可以设置为true,重启之后,管理员密码将改成123456,然后请及时修改回false
|
# #↓↓↓↓ ----------------------------- 如果忘记管理员密码,可以设置为true,docker compose up -d 重建容器之后,管理员密码将改成123456,然后请及时修改回false
|
||||||
- certd_system_resetAdminPasswd=false
|
- certd_system_resetAdminPasswd=false
|
||||||
|
# ↓↓↓ 要使用ipv6,将此配置修改为::
|
||||||
|
- certd_koa_hostname=0.0.0.0
|
||||||
|
|
||||||
# 默认使用sqlite文件数据库,如果需要使用其他数据库,请设置以下环境变量
|
# 默认使用sqlite文件数据库,如果需要使用其他数据库,请设置以下环境变量
|
||||||
# 注意: 选定使用一种数据库之后,不支持更换数据库。
|
# 注意: 选定使用一种数据库之后,不支持更换数据库。
|
||||||
@@ -53,7 +62,7 @@ services:
|
|||||||
# - certd_typeorm_dataSource_default_password=yourpasswd # 密码
|
# - certd_typeorm_dataSource_default_password=yourpasswd # 密码
|
||||||
# - certd_typeorm_dataSource_default_database=certd # 数据库名
|
# - certd_typeorm_dataSource_default_database=certd # 数据库名
|
||||||
|
|
||||||
# #↓↓↓↓ ----------------------------- 使用mysql数据库,需要提前创建数据库 charset=utf8mb4, collation=utf8mb4_bin
|
# #↓↓↓↓ ----------------------------- 使用mysql8数据库,需要提前创建数据库 charset=utf8mb4, collation=utf8mb4_bin
|
||||||
# - certd_flyway_scriptDir=./db/migration-mysql # 升级脚本目录
|
# - certd_flyway_scriptDir=./db/migration-mysql # 升级脚本目录
|
||||||
# - certd_typeorm_dataSource_default_type=mysql # 数据库类型, 或者 mariadb
|
# - certd_typeorm_dataSource_default_type=mysql # 数据库类型, 或者 mariadb
|
||||||
# - certd_typeorm_dataSource_default_host=localhost # 数据库地址
|
# - certd_typeorm_dataSource_default_host=localhost # 数据库地址
|
||||||
|
|||||||
@@ -81,42 +81,57 @@ export default defineConfig({
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
{text: "演示教程", link: "/guide/tutorial.md"},
|
{text: "演示教程", link: "/guide/tutorial.md"},
|
||||||
{text: "版本升级", link: "/guide/install/upgrade.md"}
|
{text: "版本升级", link: "/guide/install/upgrade.md"},
|
||||||
|
{text: "赞助专业版", link: "/guide/donate/"},
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: "特性",
|
text: "特性",
|
||||||
items: [
|
items: [
|
||||||
{text: "CNAME代理校验", link: "/guide/feature/cname/index.md"},
|
{text: "CNAME代理校验", link: "/guide/feature/cname/index.md"},
|
||||||
{text: "插件列表", link: "/guide/plugins.md"},
|
|
||||||
{text: "多数据库支持", link: "/guide/install/database.md"},
|
{text: "多数据库支持", link: "/guide/install/database.md"},
|
||||||
{text: "开放接口", link: "/guide/open/index.md"},
|
{text: "开放接口", link: "/guide/open/index.md"},
|
||||||
{
|
{
|
||||||
text: "站点安全", items: [
|
text: "站点安全", link: "/guide/feature/safe/"
|
||||||
{text: "安全特性", link: "/guide/feature/safe"},
|
},
|
||||||
{text: "站点隐藏", link: "/guide/feature/safe/hidden"},
|
{
|
||||||
{text: "安全生产建议", link: "/guide/feature/safe/suggest"},
|
text: "插件列表", items: [
|
||||||
|
{text: "DNS提供商", link: "/guide/plugins/dns-provider.md"},
|
||||||
|
{text: "任务插件", link: "/guide/plugins/deploy.md"},
|
||||||
|
{text: "通知插件", link: "/guide/plugins/notification.md"},
|
||||||
|
{text: "授权提供商", link: "/guide/plugins/access.md"},
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: "常见问题",
|
text: "常见问题",
|
||||||
items: [
|
items: [
|
||||||
|
{text: "QA", link: "/guide/qa/use.md"},
|
||||||
|
{text: "忘记密码/无法登录", link: "/guide/use/forgotpasswd/"},
|
||||||
{text: "群晖证书部署", link: "/guide/use/synology/"},
|
{text: "群晖证书部署", link: "/guide/use/synology/"},
|
||||||
{text: "腾讯云密钥获取", link: "/guide/use/tencent/"},
|
{text: "腾讯云密钥获取", link: "/guide/use/tencent/"},
|
||||||
{text: "连接windows主机", link: "/guide/use/host/windows.md"},
|
{text: "连接windows主机", link: "/guide/use/host/windows.md"},
|
||||||
{text: "Google EAB获取", link: "/guide/use/google/"},
|
{text: "Google EAB获取", link: "/guide/use/google/"},
|
||||||
{text: "阿里云相关", link: "/guide/use/aliyun/"},
|
{text: "阿里云相关", link: "/guide/use/aliyun/"},
|
||||||
{text: "忘记密码", link: "/guide/use/forgotpasswd/"},
|
|
||||||
{text: "数据备份", link: "/guide/use/backup/"},
|
{text: "数据备份", link: "/guide/use/backup/"},
|
||||||
{text: "Certd本身的证书更新", link: "/guide/use/https/index.md"},
|
{text: "Certd本身的证书更新", link: "/guide/use/https/index.md"},
|
||||||
{text: "js脚本插件使用", link: "/guide/use/custom-script/index.md"},
|
{text: "js脚本插件使用", link: "/guide/use/custom-script/index.md"},
|
||||||
{text: "邮箱配置", link: "/guide/use/email/index.md"},
|
{text: "邮箱配置", link: "/guide/use/email/index.md"},
|
||||||
{text: "IPv6支持", link: "/guide/use/setting/ipv6.md"},
|
{text: "IPv6支持", link: "/guide/use/setting/ipv6.md"},
|
||||||
{text: "其他插件使用", link: "/deploy/"},
|
{text: "ESXi", link: "/guide/use/ESXi/index.md"},
|
||||||
{text: "商业版说明", link: "/comm/"},
|
{text: "宝塔动态IP白名单", link: "/guide/use/baota/white_list.md"},
|
||||||
|
{text: "子域名托管", link: "/guide/use/cert/subdomain.md"},
|
||||||
|
{text: "流水线有效期", link: "/guide/use/pipeline/valid.md"},
|
||||||
|
{text: "IP证书申请", link: "/guide/use/cert/ip.md"},
|
||||||
|
{text: "插件开发", link: "/guide/use/dev/plugin.md"},
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: "商业版配置", link: "/guide/use/comm/", items: [
|
||||||
|
{text: "支付宝配置", link: "/guide/use/comm/payments/alipay.md"},
|
||||||
|
{text: "微信支付配置", link: "/guide/use/comm/payments/wxpay.md"},
|
||||||
|
{text: "彩虹易支付配置", link: "/guide/use/comm/payments/yizhifu.md"},
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -126,33 +141,11 @@ export default defineConfig({
|
|||||||
{text: "更新日志", link: "/guide/changelogs/CHANGELOG.md"},
|
{text: "更新日志", link: "/guide/changelogs/CHANGELOG.md"},
|
||||||
{text: "镜像说明", link: "/guide/image.md"},
|
{text: "镜像说明", link: "/guide/image.md"},
|
||||||
{text: "联系我们", link: "/guide/contact/"},
|
{text: "联系我们", link: "/guide/contact/"},
|
||||||
{text: "捐赠", link: "/guide/donate/"},
|
|
||||||
{text: "开源协议", link: "/guide/license/"},
|
{text: "开源协议", link: "/guide/license/"},
|
||||||
{text: "我的其他开源项目", link: "/guide/link/"},
|
{text: "我的其他开源项目", link: "/guide/link/"},
|
||||||
|
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"/deploy/": [
|
|
||||||
{
|
|
||||||
text: "部署证书插件",
|
|
||||||
items: [
|
|
||||||
{text: "插件说明", link: "/deploy/index.md"},
|
|
||||||
{text: "部署到ESXi", link: "/deploy/ESXi/index.md"},
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"/comm/": [
|
|
||||||
{
|
|
||||||
text: "商业版",
|
|
||||||
items: [
|
|
||||||
{text: "支付宝配置", link: "/comm/payments/alipay.md"},
|
|
||||||
{text: "微信支付配置", link: "/comm/payments/wxpay.md"},
|
|
||||||
{text: "彩虹易支付配置", link: "/comm/payments/yizhifu.md"},
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
,
|
|
||||||
},
|
},
|
||||||
|
|
||||||
socialLinks: [
|
socialLinks: [
|
||||||
|
|||||||
@@ -1,4 +0,0 @@
|
|||||||
# 部署插件说明
|
|
||||||
|
|
||||||
## 待完善
|
|
||||||
|
|
||||||
@@ -0,0 +1,88 @@
|
|||||||
|
|
||||||
|
# 授权插件Demo
|
||||||
|
|
||||||
|
```ts
|
||||||
|
import { AccessInput, BaseAccess, IsAccess } from '@certd/pipeline';
|
||||||
|
import { isDev } from '../../utils/env.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 这个注解将注册一个授权配置
|
||||||
|
* 在certd的后台管理系统中,用户可以选择添加此类型的授权
|
||||||
|
*/
|
||||||
|
@IsAccess({
|
||||||
|
name: 'demo',
|
||||||
|
title: '授权插件示例',
|
||||||
|
icon: 'clarity:plugin-line',
|
||||||
|
desc: '',
|
||||||
|
})
|
||||||
|
export class DemoAccess extends BaseAccess {
|
||||||
|
/**
|
||||||
|
* 授权属性配置
|
||||||
|
*/
|
||||||
|
@AccessInput({
|
||||||
|
title: '密钥Id',
|
||||||
|
component: {
|
||||||
|
placeholder: 'demoKeyId',
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
demoKeyId = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 授权属性配置
|
||||||
|
*/
|
||||||
|
@AccessInput({
|
||||||
|
//标题
|
||||||
|
title: '密钥串',
|
||||||
|
component: {
|
||||||
|
//input组件的placeholder
|
||||||
|
placeholder: 'demoKeySecret',
|
||||||
|
},
|
||||||
|
//是否必填
|
||||||
|
required: true,
|
||||||
|
//改属性是否需要加密
|
||||||
|
encrypt: true,
|
||||||
|
})
|
||||||
|
//属性名称
|
||||||
|
demoKeySecret = '';
|
||||||
|
}
|
||||||
|
new DemoAccess();
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
# 阿里云授权
|
||||||
|
```ts
|
||||||
|
|
||||||
|
import { IsAccess, AccessInput, BaseAccess } from "@certd/pipeline";
|
||||||
|
|
||||||
|
@IsAccess({
|
||||||
|
name: "aliyun",
|
||||||
|
title: "阿里云授权",
|
||||||
|
desc: "",
|
||||||
|
icon: "ant-design:aliyun-outlined",
|
||||||
|
order: 0,
|
||||||
|
})
|
||||||
|
export class AliyunAccess extends BaseAccess {
|
||||||
|
@AccessInput({
|
||||||
|
title: "accessKeyId",
|
||||||
|
component: {
|
||||||
|
placeholder: "accessKeyId",
|
||||||
|
},
|
||||||
|
helper: "登录阿里云控制台->AccessKey管理页面获取。",
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
accessKeyId = "";
|
||||||
|
@AccessInput({
|
||||||
|
title: "accessKeySecret",
|
||||||
|
component: {
|
||||||
|
placeholder: "accessKeySecret",
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
encrypt: true,
|
||||||
|
helper: "注意:证书申请需要dns解析权限;其他阿里云插件,需要对应的权限,比如证书上传需要证书管理权限;嫌麻烦就用主账号的全量权限的accessKey",
|
||||||
|
})
|
||||||
|
accessKeySecret = "";
|
||||||
|
}
|
||||||
|
|
||||||
|
new AliyunAccess();
|
||||||
|
```
|
||||||
|
After Width: | Height: | Size: 56 KiB |
|
Before Width: | Height: | Size: 130 KiB After Width: | Height: | Size: 76 KiB |
|
After Width: | Height: | Size: 305 KiB |
|
After Width: | Height: | Size: 92 KiB |
@@ -1,28 +1,56 @@
|
|||||||
# 捐赠
|
# 专业版赞助
|
||||||
************************
|
|
||||||
支持开源,为爱发电,我已入驻爱发电
|
|
||||||
https://afdian.com/a/greper
|
|
||||||
|
|
||||||
## 发电权益:
|
## 开源为什么要做专业版收费?
|
||||||
1. 可加入发电专属群,可以获得作者一对一技术支持
|
1. 纯靠为爱发电不可持续,容易烂尾(比如:我的[dev-sidecar项目](https://github.com/docmirror/dev-sidecar)即便是拥有20K+star,也差点凉凉,幸亏有另外大佬接手用爱发电)
|
||||||
|
2. 没有赞助的项目,作者会比较任性,不会用心倾听用户的心声,不顾用户体验(比如:下意识拒绝需求、频繁破坏性变更升级、全盘推倒重来之类的)
|
||||||
|
3. 没有赞助的项目,交流群的戾气有时候比较重,容易起冲突
|
||||||
|
|
||||||
|
## 赞助权益:
|
||||||
|
1. 可加入专属VIP群,可以获得作者一对一技术支持,必要时可以远程协助
|
||||||
2. 您的需求我们将优先实现,并且将作为专业版功能提供
|
2. 您的需求我们将优先实现,并且将作为专业版功能提供
|
||||||
3. 一年期专业版激活码
|
3. 获得专业版功能
|
||||||
|
|
||||||
|
****------------------****
|
||||||
|
> [限时¥50永久专业版优惠券,点我立刻领取](https://app.handfree.work/subject/#/app/certd/product)
|
||||||
|
|
||||||
|
****------------------****
|
||||||
## 专业版特权对比
|
## 专业版特权对比
|
||||||
|
|
||||||
| 功能 | 免费版 | 专业版 |
|
| 功能 | 免费版 | 专业版 |
|
||||||
|---------|------------------------|-----------------------------|
|
|---------|---------------------------------------|--------------------------------|
|
||||||
| 免费证书申请 | 免费无限制 | 免费无限制 |
|
| 证书申请 | 无限制 | 无限制 |
|
||||||
| 自动部署插件 | 阿里云CDN、腾讯云、七牛CDN、主机部署等 | 支持群晖、宝塔、1Panel等,持续开发中 |
|
| 证书域名数量 | 无限制 | 无限制 |
|
||||||
| 证书流水线条数 | 无限制 | 无限制 |
|
| 证书流水线条数 | 无限制 | 无限制 |
|
||||||
| 站点证书监控 | 限制1条 | 无限制 |
|
| 自动部署插件 | 阿里云CDN、腾讯云、七牛CDN、主机部署、宝塔、1Panel等大部分插件 | 群晖、威联通、proxmox等 |
|
||||||
| 通知 | 邮件通知、自定义webhook | 邮件免配置、企微、飞书、anpush、server酱等 |
|
| 通知 | 邮件通知、自定义webhook | 邮件免配置、企微、钉钉、飞书、anpush、server酱等 |
|
||||||
|
| 站点监控 | 限制1条 | 无限制 |
|
||||||
|
| 批量操作 | 无 | 流水线模版,流水线复制,批量运行,批量设置通知、定时等 |
|
||||||
|
| VIP群 | 无 | 可加,一对一技术支持,必要时可申请远程协助 |
|
||||||
|
|
||||||
|
|
||||||
## 专业版激活方式
|
## 专业版激活方式
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
发电后,在私信中获取激活码
|
|
||||||
************************
|
## 相关问题
|
||||||
|
|
||||||
|
### 1. 购买后VIP状态或时长未更新
|
||||||
|
系统管理-->账号绑定页面,打开一下即可自动更新VIP最新状态(如果未登录袖手账号需要先登录)
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
### 2. 开发票
|
||||||
|
联系我们(微信:xiaojunnuo),并提供支付金额
|
||||||
|
|
||||||
|
### 3. VIP是否可以迁移换绑服务器?
|
||||||
|
可以的。
|
||||||
|
* 方式1. 直接将备份数据在新服务器上还原即可(首次访问会提示您是否绑定新url,点击是即可)
|
||||||
|
* 方式2. 如果旧站点数据丢失,您也可以部署一个新站点,然后在系统管理-->账号绑定页面,转移VIP即可
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -22,4 +22,6 @@
|
|||||||

|

|
||||||
|
|
||||||
## 3、忘记解除地址和解除密码怎么办
|
## 3、忘记解除地址和解除密码怎么办
|
||||||
登录服务器,在数据库平级的目录下创建`.unhidden`文件即可`临时解除`站点隐藏
|
登录服务器,在数据库平级的目录下创建`.unhidden`命名的空白文件,即可临时解除站点隐藏
|
||||||
|
临时解除后会自动删除`.unhidden`文件,请尽快设置好新的`解除地址`和`解除密码`,并记住
|
||||||
|
|
||||||
|
|||||||
|
After Width: | Height: | Size: 70 KiB |
@@ -1,34 +1,49 @@
|
|||||||
# 站点安全特性
|
# 安全特性
|
||||||
|
|
||||||
Certd 存储了证书以及授权等敏感数据,所以需要严格保障安全。
|
Certd 存储了证书以及授权等敏感数据,所以需要严格保障安全。
|
||||||
我们非常重视您的数据安全,提供了以下安全特性
|
我们提供了以下安全特性,以及安全生产建议(请遵照建议进行生产部署以保障数据安全)
|
||||||
|
|
||||||
## 1、 授权数据加密存储【默认开启】
|
## 一、站点安全特性
|
||||||
|
|
||||||
|
### 1、 授权数据加密存储【默认开启】
|
||||||
* 所有的授权敏感字段会加密后存储
|
* 所有的授权敏感字段会加密后存储
|
||||||
* 每个用户独立维护授权数据,连管理员都无权查看
|
* 每个用户独立维护授权数据,连管理员都无权查看
|
||||||
|
|
||||||

|

|
||||||
星号部分为加密数据
|
星号部分为加密数据
|
||||||
|
|
||||||
## 2、 密码防爆破【默认开启】
|
### 2、 密码防爆破【默认开启】
|
||||||
* 登录失败次数过多,账号将被锁定,最高24小时(重启服务可解除锁定)
|
* 登录失败次数过多,账号将被锁定,最高24小时(重启服务可解除锁定)
|
||||||
* 用户登录密码加密hash后存储,无法计算出密码明文
|
* 用户登录密码加密hash后存储,无法计算出密码明文
|
||||||

|

|
||||||
|
|
||||||
## 3、站点隐藏【建议开启】
|
### 3、站点隐藏【建议开启】
|
||||||
* 一般来说Certd设置好之后,后续很少需要访问修改。
|
* 一般来说Certd设置好之后,后续很少需要访问修改。
|
||||||
* 所以我们平时可以把站点访问关闭,需要的时候再打开,减少站点被攻击的风险
|
* 所以我们平时可以把站点访问关闭,需要的时候再打开,减少站点被攻击的风险
|
||||||
* 请前往 `系统管理->系统设置->安全设置->开启站点隐藏`
|
* 请前往 `系统管理->系统设置->安全设置->开启站点隐藏`
|
||||||
* [站点隐藏设置说明](./hidden/)
|
|
||||||

|

|
||||||
|
|
||||||
## 4、登录二次验证
|
点击查看 [站点隐藏功能详细使用说明](./hidden/)
|
||||||
|
|
||||||
待实现
|
|
||||||
|
|
||||||
## 5、数据库自动备份【建议开启】
|
### 4、登录双重验证
|
||||||
|
|
||||||
|
支持2FA双重认证
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
### 5、数据库自动备份【建议开启】
|
||||||
* [自动备份设置说明](../../use/backup/)
|
* [自动备份设置说明](../../use/backup/)
|
||||||
|
|
||||||
|
|
||||||
## 更多安全生产建议
|
## 二、安全生产建议
|
||||||
[安全生产建议](./suggest.md)
|
|
||||||
|
尽管`Cert`本身实现了很多安全特性,但`外部环境的安全`仍需要您来确保。
|
||||||
|
请`务必`遵循如下建议做好安全防护
|
||||||
|
|
||||||
|
* 请`务必`使用`HTTPS协议`访问本应用,避免被中间人攻击
|
||||||
|
* 请`务必`使用`web应用防火墙`防护本应用,防止XSS、SQL注入等攻击
|
||||||
|
* 请`务必`做好`服务器本身`的安全防护,防止数据库泄露
|
||||||
|
* 请`务必`做好[`数据备份`](../../use/backup/),避免数据丢失
|
||||||
|
* 请`务必`修改管理员账号用户名,且建议将admin注册为普通用户,且设置为禁用。
|
||||||
|
* 建议开启[`站点隐藏`](./hidden/)功能
|
||||||
|
|||||||
@@ -1,10 +0,0 @@
|
|||||||
# 安全生产建议
|
|
||||||
|
|
||||||
尽管`Cert`本身实现了很多安全特性,但`外部环境的安全`仍需要您来确保。
|
|
||||||
请`务必`遵循如下建议做好安全防护
|
|
||||||
|
|
||||||
* 请`务必`使用`HTTPS协议`访问本应用,避免被中间人攻击
|
|
||||||
* 请`务必`使用`web应用防火墙`防护本应用,防止XSS、SQL注入等攻击
|
|
||||||
* 请`务必`做好`服务器本身`的安全防护,防止数据库泄露
|
|
||||||
* 请`务必`做好[`数据备份`](../../use/backup/),避免数据丢失
|
|
||||||
* 建议开启[`站点隐藏`](./hidden/)功能
|
|
||||||
|
Before Width: | Height: | Size: 26 KiB |
@@ -6,29 +6,39 @@ Certd 是一款开源、免费、全自动申请和部署更新SSL证书的工
|
|||||||
关键字:证书自动申请、证书自动更新、证书自动续期、证书自动续签、证书管理工具
|
关键字:证书自动申请、证书自动更新、证书自动续期、证书自动续签、证书管理工具
|
||||||
|
|
||||||
|
|
||||||
## 一、特性
|
| 官方开源地址: | |
|
||||||
|
| ---- | ---- |
|
||||||
|
| [Github](https://github.com/certd/certd)|  |
|
||||||
|
| [Gitee](https://gitee.com/certd/certd) |  |
|
||||||
|
| [AtomGit](https://atomgit.com/certd/certd) | |
|
||||||
|
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## 1、关于证书续期
|
||||||
|
>* 实际上没有办法不改变证书文件本身情况下直接续期或者续签。
|
||||||
|
>* 我们所说的续期,其实就是按照全套流程重新申请一份新证书,然后重新部署上去。
|
||||||
|
>* 免费证书过期时间90天,以后可能还会缩短,所以自动化部署必不可少
|
||||||
|
|
||||||
|
|
||||||
|
## 2、项目特性
|
||||||
本项目不仅支持证书申请过程自动化,还可以自动化部署更新证书,让你的证书永不过期。
|
本项目不仅支持证书申请过程自动化,还可以自动化部署更新证书,让你的证书永不过期。
|
||||||
|
|
||||||
* 全自动申请证书(支持所有注册商注册的域名)
|
* 全自动申请证书(支持所有注册商注册的域名,支持DNS-01、HTTP-01、CNAME代理等多种域名验证方式)
|
||||||
* 全自动部署更新证书(目前支持部署到主机、部署到阿里云、腾讯云等,目前已支持60+部署插件)
|
* 全自动部署更新证书(目前支持部署到主机、阿里云、腾讯云等100+部署插件)
|
||||||
* 支持通配符域名/泛域名,支持多个域名打到一个证书上
|
* 支持通配符域名/泛域名,支持多个域名打到一个证书上,支持pem、pfx、der、jks等多种证书格式
|
||||||
* 邮件通知
|
* 邮件通知、webhook通知、企微、钉钉、飞书、anpush等多种通知方式
|
||||||
* 私有化部署,保障数据安全
|
* 私有化部署,数据保存本地,安装升级非常简单快捷
|
||||||
* 支持SQLite、Postgresql、MySQL数据库
|
* 镜像由Github Actions构建,过程公开透明
|
||||||
|
* 授权加密,站点隐藏,2FA,密码防爆破等多重安全保障
|
||||||
|
* 支持SQLite,PostgreSQL、MySQL多种数据库
|
||||||
|
* 开放接口支持
|
||||||
|
* 站点证书监控
|
||||||
|
* 多用户管理
|
||||||
|
|
||||||
|
|
||||||
## 二、一些说明
|

|
||||||
* 本项目申请证书过程遵循acme协议
|
|
||||||
* 需要验证域名所有权,一般有两种方式
|
|
||||||
* http-01: 在网站根目录下放置一份txt文件
|
|
||||||
* dns-01: 需要给域名添加txt解析记录,通配符域名只能用这种方式(本项目仅支持dns-01)
|
|
||||||
* 证书续期:
|
|
||||||
* 实际上没有办法不改变证书文件本身情况下直接续期或者续签。
|
|
||||||
* 我们所说的续期,其实就是按照全套流程重新申请一份新证书,然后重新部署上去。
|
|
||||||
* 免费证书过期时间90天,以后可能还会缩短,所以自动化部署必不可少
|
|
||||||
* 设置每天自动运行,当证书过期前35天,会自动重新申请证书并部署
|
|
||||||
|
|
||||||
## 三、证书颁发机构对比
|
|
||||||
* Let's Encrypt:申请最简单。
|
|
||||||
* Google: 大厂光环,兼容性好,首次需要翻墙获取EAB。
|
|
||||||
* ZeroSSL: 需要EAB,获取EAB无需翻墙。
|
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
{
|
||||||
|
"notice": "永久专业版上线,新用户立减50,升级到最新版点击下方“立即赞助”按钮前往获取",
|
||||||
|
"plus": {
|
||||||
|
"name": "专业版",
|
||||||
|
"price": "89.9",
|
||||||
|
"price3": "199",
|
||||||
|
"tooltip": "开源需要您的赞助支持",
|
||||||
|
"priceText":"¥89.9/年",
|
||||||
|
"discountText":"永久专业版50优惠券立即领取"
|
||||||
|
},
|
||||||
|
"comm": {
|
||||||
|
"name": "商业版",
|
||||||
|
"price": "399",
|
||||||
|
"price3": "899",
|
||||||
|
"tooltip": "3年优惠300",
|
||||||
|
"priceText":"¥399/年",
|
||||||
|
"discountText":"¥899/3年(3年优惠300)"
|
||||||
|
},
|
||||||
|
"app":{
|
||||||
|
"minVersion":"1.36.0",
|
||||||
|
"minVersionTip":"版本过低,为了您的数据安全,请尽快升级"
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
After Width: | Height: | Size: 82 KiB |
|
After Width: | Height: | Size: 49 KiB |
|
After Width: | Height: | Size: 21 KiB |
@@ -7,7 +7,36 @@ https://1panel.cn/docs/installation/online_installation/
|
|||||||
|
|
||||||
## 二、部署certd
|
## 二、部署certd
|
||||||
|
|
||||||
|
有两种安装方式
|
||||||
|
|
||||||
|
### 1. 应用商店方式安装【推荐】
|
||||||
|
|
||||||
|
#### 1.1 安装
|
||||||
|
打开`1Panel->应用商店`,更新远程应用,搜索`certd`,点击安装
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
#### 1.2 访问测试:
|
||||||
|
http://ip:7001
|
||||||
|
https://ip:7002
|
||||||
|
默认账号密码
|
||||||
|
admin/123456
|
||||||
|
登录后请及时修改密码
|
||||||
|
|
||||||
|
#### 1.3 备份
|
||||||
|

|
||||||
|
|
||||||
|
#### 1.4 恢复
|
||||||
|
安装新Certd后,点击导入备份按钮,选择上面备份的文件即可
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### 2. docker-compose方式安装
|
||||||
|
|
||||||
|
#### 2.1 安装
|
||||||
1. 打开`docker-compose.yaml`,整个内容复制下来
|
1. 打开`docker-compose.yaml`,整个内容复制下来
|
||||||
https://gitee.com/certd/certd/raw/v2/docker/run/docker-compose.yaml
|
https://gitee.com/certd/certd/raw/v2/docker/run/docker-compose.yaml
|
||||||
|
|
||||||
@@ -22,7 +51,7 @@ https://1panel.cn/docs/installation/online_installation/
|
|||||||
> 默认使用sqlite数据库,数据保存在`/data/certd`目录下,您可以手动备份该目录
|
> 默认使用sqlite数据库,数据保存在`/data/certd`目录下,您可以手动备份该目录
|
||||||
> certd还支持`mysql`和`postgresql`数据库,[点我了解如何切换其他数据库](../database)
|
> certd还支持`mysql`和`postgresql`数据库,[点我了解如何切换其他数据库](../database)
|
||||||
|
|
||||||
3. 访问测试
|
#### 2.2 访问测试
|
||||||
|
|
||||||
http://ip:7001
|
http://ip:7001
|
||||||
https://ip:7002
|
https://ip:7002
|
||||||
@@ -30,7 +59,7 @@ https://ip:7002
|
|||||||
admin/123456
|
admin/123456
|
||||||
登录后请及时修改密码
|
登录后请及时修改密码
|
||||||
|
|
||||||
## 三、升级
|
#### 2.3 升级
|
||||||
|
|
||||||
1. 找到容器,点击更多->升级
|
1. 找到容器,点击更多->升级
|
||||||

|

|
||||||
@@ -39,11 +68,11 @@ admin/123456
|
|||||||

|

|
||||||
|
|
||||||
|
|
||||||
## 四、数据备份
|
#### 2.4 备份
|
||||||
|
|
||||||
> 默认数据保存在`/data/certd`目录下,可以手动备份
|
> 默认数据保存在`/data/certd`目录下,可以手动备份
|
||||||
> 建议配置一条 [数据库备份流水线](../../use/backup/),自动备份
|
> 建议配置一条 [数据库备份流水线](../../use/backup/),自动备份
|
||||||
|
|
||||||
## 五、备份恢复
|
#### 2.5 恢复
|
||||||
|
|
||||||
将备份的`db.sqlite`及同目录下的其他文件一起覆盖到原来的位置,重启certd即可
|
将备份的`db.sqlite`及同目录下的其他文件一起覆盖到原来的位置,重启certd即可
|
||||||
|
|||||||
|
After Width: | Height: | Size: 82 KiB |
@@ -10,23 +10,24 @@
|
|||||||
* 登录宝塔面板,在菜单栏中点击 Docker,首次进入会提示安装Docker服务,点击立即安装,按提示完成安装
|
* 登录宝塔面板,在菜单栏中点击 Docker,首次进入会提示安装Docker服务,点击立即安装,按提示完成安装
|
||||||
|
|
||||||
### 2、部署certd
|
### 2、部署certd
|
||||||
|
以下两种方式任选一种:
|
||||||
|
|
||||||
#### 2.1 应用商店一键部署【推荐】
|
#### 2.1 应用商店方式一键部署【推荐】
|
||||||
|
|
||||||
* 在应用商店中找到`certd`(要先点右上角更新应用)
|
* 在宝塔Docker应用商店中找到`certd`(要先点右上角更新应用)
|
||||||
* 点击安装,配置域名等基本信息即可完成安装
|
* 点击安装,配置域名等基本信息即可完成安装
|
||||||
|
|
||||||
> 需要宝塔9.2.0及以上版本才支持
|
> 需要宝塔9.2.0及以上版本才支持
|
||||||
|
|
||||||
#### 2.2 容器编排部署
|
#### 2.2 容器编排方式部署
|
||||||
|
|
||||||
1. 打开`docker-compose.yaml`,整个内容复制下来
|
1. 打开`docker-compose.yaml`,整个内容复制下来
|
||||||
https://gitee.com/certd/certd/raw/v2/docker/run/docker-compose.yaml
|
https://gitee.com/certd/certd/raw/v2/docker/run/docker-compose.yaml
|
||||||
|
|
||||||
|
|
||||||
然后到宝塔里面进到docker->容器编排->添加容器编排
|
然后到宝塔里面进到docker->容器编排->添加容器编排
|
||||||

|

|
||||||
点击确定,等待启动完成
|
点击确定,等待启动完成
|
||||||

|

|
||||||
|
|
||||||
> certd默认使用sqlite数据库,另外支持`mysql`和`postgresql`数据库,[点我了解如何切换其他数据库](../database)
|
> certd默认使用sqlite数据库,另外支持`mysql`和`postgresql`数据库,[点我了解如何切换其他数据库](../database)
|
||||||
@@ -34,21 +35,24 @@
|
|||||||
|
|
||||||
## 二、访问应用
|
## 二、访问应用
|
||||||
|
|
||||||
http://ip:7001
|
http://ip:7001
|
||||||
https://ip:7002
|
https://ip:7002
|
||||||
默认账号密码
|
默认账号密码
|
||||||
admin/123456
|
admin/123456
|
||||||
登录后请及时修改密码
|
登录后请及时修改密码
|
||||||
|
|
||||||
## 三、如何升级
|
## 三、如何升级
|
||||||
宝塔升级certd非常简单
|
宝塔升级certd非常简单
|
||||||
|
|
||||||
`docker`->`容器编排`->`左侧选择Certd`->`更新镜像`
|
打开容器页面: `docker`->`容器编排`->`左侧选择Certd`->`更新镜像`
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
|
|
||||||
## 四、数据备份
|
## 四、数据备份
|
||||||
|
|
||||||
|
部署方式不同,数据保存位置不同
|
||||||
|
|
||||||
### 4.1 应用商店部署方式
|
### 4.1 应用商店部署方式
|
||||||
点击进入安装路径,数据保存在`./data`目录下,可以手动备份
|
点击进入安装路径,数据保存在`./data`目录下,可以手动备份
|
||||||
|
|
||||||
@@ -62,7 +66,6 @@ admin/123456
|
|||||||
数据默认保存在`/data/certd`目录下,可以手动备份
|
数据默认保存在`/data/certd`目录下,可以手动备份
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### 4.3 自动备份
|
### 4.3 自动备份
|
||||||
|
|
||||||
> 建议配置一条 [数据库备份流水线](../../use/backup/),自动备份
|
> 建议配置一条 [数据库备份流水线](../../use/backup/),自动备份
|
||||||
@@ -70,3 +73,15 @@ admin/123456
|
|||||||
## 五、备份恢复
|
## 五、备份恢复
|
||||||
|
|
||||||
将备份的`db.sqlite`及同目录下的其他文件一起覆盖到原来的位置,重启certd即可
|
将备份的`db.sqlite`及同目录下的其他文件一起覆盖到原来的位置,重启certd即可
|
||||||
|
|
||||||
|
|
||||||
|
## 六、宝塔部署相关问题排查
|
||||||
|
|
||||||
|
### 1. 无法访问Certd
|
||||||
|
1. 确认服务器的安全规则,是否放开了对应端口
|
||||||
|
2. 确认宝塔防火墙是否放开对应端口
|
||||||
|
3. 尝试将Certd容器加入宝塔的`bridge`网络
|
||||||
|

|
||||||
|
|
||||||
|
### 2. 动态IP无法加白名单问题
|
||||||
|
[Nginx代理解决方案](../../use/baota/white_list.md)
|
||||||
@@ -65,9 +65,54 @@ docker-compose up -d
|
|||||||
|
|
||||||
## 二、从旧版的sqlite切换数据库
|
## 二、从旧版的sqlite切换数据库
|
||||||
|
|
||||||
1. 先将`旧certd`升级到最新版 (`建议:备份sqlite数据库` )
|
从旧版`sqlite`迁移到`mysql`或`postgresql`数据库
|
||||||
2. 按照上面全新安装方式部署一套`新的certd` (`注意:新旧版本的certd要一致`)
|
|
||||||
3. 使用数据库工具将数据从sqlite导入到mysql或postgresql (`注意:flyway_history数据表不要导入`)
|
|
||||||
4. 重启新certd
|
|
||||||
5. 确认没有问题之后,删除旧版certd
|
|
||||||
|
|
||||||
|
下面以 `SQLite` 转 `MySQL` 为例进行演示
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
#### 0.前提条件:
|
||||||
|
1. SQLite版Certd站点已经`升级到最新版` (`建议:备份sqlite数据库` )
|
||||||
|
2. `全新安装`MySQL版本Certd(`确保是全新的,因为里面的数据会被清空覆盖`)
|
||||||
|
3. 两套Certd站点版本一致
|
||||||
|
|
||||||
|
#### 1. 安装DBeaver工具
|
||||||
|
|
||||||
|
[https://dbeaver.io/download/](https://dbeaver.io/download/)
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
#### 2. 连接到sqlite数据库
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
#### 3. 连接到mysql或postgresql数据库
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
#### 4. 开始同步数据
|
||||||
|
|
||||||
|
选择mysql数据库,选择所有的表(`flyway_history除外`),右键导入数据
|
||||||
|
|
||||||
|
> 切记flyway_history数据表不要导入
|
||||||
|
|
||||||
|

|
||||||
|

|
||||||
|

|
||||||
|
下一步、下一步,直到数据加载设置,勾选`在加载前截断目标表`(此选项很重要,并且会清空mysql certd数据库中的数据)
|
||||||
|

|
||||||
|
|
||||||
|
#### 5. 导入完成
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
#### 6. 重启MySQL版本Certd
|
||||||
|
|
||||||
|
访问MySQL版本测试,数据已成功迁移
|
||||||
|
|
||||||
|
确认没有问题之后,删除旧版certd
|
||||||
@@ -55,6 +55,15 @@ https://your_server_ip:7002
|
|||||||
|
|
||||||
## 二、升级
|
## 二、升级
|
||||||
|
|
||||||
|
::: warning
|
||||||
|
如果您是第一次升级certd版本,切记切记先备份一下数据
|
||||||
|
```
|
||||||
|
# docker-compose.yaml配置
|
||||||
|
- /data/certd:/app/data # 请务必确保 /app/data 这个路径没有改动,固定写死
|
||||||
|
```
|
||||||
|
:::
|
||||||
|
|
||||||
|
|
||||||
### 如果使用固定版本号
|
### 如果使用固定版本号
|
||||||
1. 修改`docker-compose.yaml`中的镜像版本号
|
1. 修改`docker-compose.yaml`中的镜像版本号
|
||||||
2. 运行`docker compose up -d` 即可
|
2. 运行`docker compose up -d` 即可
|
||||||
|
|||||||
|
After Width: | Height: | Size: 74 KiB |
|
After Width: | Height: | Size: 134 KiB |
|
After Width: | Height: | Size: 145 KiB |
|
After Width: | Height: | Size: 102 KiB |
|
After Width: | Height: | Size: 194 KiB |
|
After Width: | Height: | Size: 120 KiB |
|
After Width: | Height: | Size: 157 KiB |
|
After Width: | Height: | Size: 187 KiB |
|
After Width: | Height: | Size: 67 KiB |
|
After Width: | Height: | Size: 105 KiB |
|
After Width: | Height: | Size: 145 KiB |
|
After Width: | Height: | Size: 28 KiB |
|
After Width: | Height: | Size: 21 KiB |
@@ -1,17 +1,21 @@
|
|||||||
# 源码部署
|
# 源码部署
|
||||||
如果没有`git`和`nodejs`基础,则不推荐
|
如果没有开发基础、没有运维基础、没有`git`和`nodejs`基础,强烈不推荐此方式
|
||||||
|
|
||||||
## 一、源码安装
|
## 一、源码安装
|
||||||
|
|
||||||
### 环境要求
|
### 环境要求
|
||||||
- nodejs 20 及以上
|
- nodejs 22 及以上
|
||||||
### 源码启动
|
### 源码启动
|
||||||
```shell
|
```shell
|
||||||
# 克隆代码
|
# 克隆代码
|
||||||
git clone https://github.com/certd/certd --depth=1
|
git clone https://github.com/certd/certd --depth=1
|
||||||
# git checkout v1.x.x # 当v2主干分支代码无法正常启动时,可以尝试此命令,1.x.x换成最新版本号
|
# git checkout v1.x.x # 当v2主干分支代码无法正常启动时,可以尝试此命令,1.x.x换成最新版本号
|
||||||
cd certd
|
cd certd
|
||||||
|
|
||||||
# 启动服务
|
# 启动服务
|
||||||
./start.sh
|
./start.sh
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
```
|
```
|
||||||
>如果是windows,请先安装`git for windows` ,然后右键,选择`open git bash here`打开终端,再执行`./start.sh`命令
|
>如果是windows,请先安装`git for windows` ,然后右键,选择`open git bash here`打开终端,再执行`./start.sh`命令
|
||||||
@@ -20,9 +24,9 @@ cd certd
|
|||||||
|
|
||||||
### 访问测试
|
### 访问测试
|
||||||
|
|
||||||
http://your_server_ip:7001
|
http://your_server_ip:7001
|
||||||
https://your_server_ip:7002
|
https://your_server_ip:7002
|
||||||
默认账号密码:admin/123456
|
默认账号密码:admin/123456
|
||||||
记得修改密码
|
记得修改密码
|
||||||
|
|
||||||
|
|
||||||
@@ -36,19 +40,39 @@ cp -rf ./packages/ui/certd-server/data ../certd-data-backup
|
|||||||
|
|
||||||
git pull
|
git pull
|
||||||
# 如果提示pull失败,可以尝试强制更新
|
# 如果提示pull失败,可以尝试强制更新
|
||||||
# git checkout v2 -f && git pull
|
# git checkout v2 -f && git pull
|
||||||
|
|
||||||
# 先停止旧的服务,7001是certd的默认端口
|
# 先停止旧的服务,7001是certd的默认端口
|
||||||
kill -9 $(lsof -t -i:7001)
|
kill -9 $(lsof -t -i:7001)
|
||||||
# 重新编译启动
|
# 重新编译启动
|
||||||
./start.sh
|
./start.sh
|
||||||
|
|
||||||
```
|
```
|
||||||
|
::: warning
|
||||||
|
升级certd版本前,切记切记先备份一下数据
|
||||||
|
:::
|
||||||
|
|
||||||
|
|
||||||
## 三、数据备份
|
## 三、数据备份
|
||||||
> 数据默认保存在 `./packages/ui/certd-server/data` 目录下
|
> 数据默认保存在 `./packages/ui/certd-server/data` 目录下
|
||||||
> 建议配置一条[数据库备份流水线](../../use/backup/) 自动备份
|
> 建议配置一条[数据库备份流水线](../../use/backup/) 自动备份
|
||||||
|
|
||||||
|
|
||||||
## 四、备份恢复
|
## 四、备份恢复
|
||||||
|
|
||||||
将备份的`db.sqlite`及同目录下的其他文件覆盖到原来的位置,重启certd即可
|
将备份的`db.sqlite`及同目录下的其他文件覆盖到原来的位置,重启certd即可
|
||||||
|
|
||||||
|
## 六、常见问题
|
||||||
|
|
||||||
|
### 1. npm install better-sqlite3 时,提示node-gyp需要vscode环境编译
|
||||||
|
|
||||||
|
1. 首先确保node版本为22以上
|
||||||
|
2. 将下面两行加到 ~/.npmrc 里面
|
||||||
|
3. 重新install
|
||||||
|
> better_sqlite3_binary_host=https://registry.npmmirror.com/-/binary/better-sqlite3
|
||||||
|
> better_sqlite3_binary_host_mirror=https://registry.npmmirror.com/-/binary/better-sqlite3
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -8,5 +8,65 @@
|
|||||||
3. [1Panel面板方式部署升级](./1panel/#三、升级)
|
3. [1Panel面板方式部署升级](./1panel/#三、升级)
|
||||||
4. [源码方式部署](./source/#二、升级)
|
4. [源码方式部署](./source/#二、升级)
|
||||||
|
|
||||||
|
::: warning
|
||||||
|
如果您是第一次升级certd版本,切记切记先备份一下数据
|
||||||
|
很多人docker不太会配置,数据目录没有映射出来,升级导致数据丢失
|
||||||
|
```
|
||||||
|
# docker-compose.yaml配置
|
||||||
|
- /data/certd:/app/data # 请务必确保 /app/data 这个路径没有改动,固定写死
|
||||||
|
```
|
||||||
|
具体备份方法可以参考上面每种部署方式升级方法后面的备份章节
|
||||||
|
:::
|
||||||
|
|
||||||
## 升级日志
|
## 升级日志
|
||||||
|
可以查看最新版本号,以及所有版本的更新日志
|
||||||
[CHANGELOG](../changelogs/CHANGELOG.md)
|
[CHANGELOG](../changelogs/CHANGELOG.md)
|
||||||
|
|
||||||
|
|
||||||
|
## 自动升级配置
|
||||||
|
|
||||||
|
### 1. 方法一:使用watchtower监控
|
||||||
|
|
||||||
|
修改docker-compose.yaml文件增加如下配置, 使用watchtower监控自动升级
|
||||||
|
```yaml
|
||||||
|
services:
|
||||||
|
certd:
|
||||||
|
...
|
||||||
|
labels:
|
||||||
|
com.centurylinklabs.watchtower.enable: "true"
|
||||||
|
|
||||||
|
# ↓↓↓↓ --------------------------------------------------------- 自动升级,上面certd的版本号要保持为latest
|
||||||
|
certd-updater: # 添加 Watchtower 服务
|
||||||
|
image: containrrr/watchtower:latest
|
||||||
|
container_name: certd-updater
|
||||||
|
restart: unless-stopped
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
# 配置 自动更新
|
||||||
|
environment:
|
||||||
|
- WATCHTOWER_CLEANUP=true # 自动清理旧版本容器
|
||||||
|
- WATCHTOWER_INCLUDE_STOPPED=false # 不更新已停止的容器
|
||||||
|
- WATCHTOWER_LABEL_ENABLE=true # 根据容器标签进行更新
|
||||||
|
- WATCHTOWER_POLL_INTERVAL=600 # 每 10 分钟检查一次更新
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### 2. 方法二:使用Certd版本监控功能
|
||||||
|
|
||||||
|
选择Github-检查Release版本插件
|
||||||
|

|
||||||
|
按如下图填写配置
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
检测到新版本后执行宿主机升级命令:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
# 拉取最新镜像
|
||||||
|
docker pull registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest
|
||||||
|
# 升级容器命令, 替换成你自己的certd更新命令
|
||||||
|
export RESTART_CERT='sleep 10; cd ~/deploy/certd/ ; docker compose down; docker compose up -d'
|
||||||
|
# 构造一个脚本10s后在后台执行,避免容器销毁时执行太快,导致流水线任务无法结束
|
||||||
|
nohup sh -c '$RESTART_CERT' >/dev/null 2>&1 & echo '10秒后重启' && exit
|
||||||
|
```
|
||||||
@@ -5,11 +5,20 @@
|
|||||||
|
|
||||||

|

|
||||||
|
|
||||||
|
:::tip
|
||||||
|
|
||||||
|
接口key分两种权限范围:
|
||||||
|
1. 仅开放接口: 仅能访问下面`接口文档`中的接口
|
||||||
|
2. 用户级别: 可访问Certd所有接口,没有文档,可以在浏览器中F12抓取网络请求参考
|
||||||
|
|
||||||
|
:::
|
||||||
|
|
||||||
## 接口文档
|
## 接口文档
|
||||||
|
|
||||||
https://apifox.com/apidoc/shared-2e76f8c4-7c58-413b-a32d-a1316529af44/254949529e0
|
https://apifox.com/apidoc/shared-2e76f8c4-7c58-413b-a32d-a1316529af44/254949529e0
|
||||||
|
|
||||||
## Token生成方法
|
|
||||||
|
### Token生成方法
|
||||||
|
|
||||||
header中传入x-certd-token即可调用开放接口
|
header中传入x-certd-token即可调用开放接口
|
||||||
1、首先从OpenKey页面生成keyId,keySecret;
|
1、首先从OpenKey页面生成keyId,keySecret;
|
||||||
@@ -17,5 +26,26 @@ header中传入x-certd-token即可调用开放接口
|
|||||||
3、将content加上keySecret进行签名: sign = md5(content + keySecret)
|
3、将content加上keySecret进行签名: sign = md5(content + keySecret)
|
||||||
4、然后将content和sign分别base64后用.号连接: x-certd-token = base64(content) +"."+base64(sign)
|
4、然后将content和sign分别base64后用.号连接: x-certd-token = base64(content) +"."+base64(sign)
|
||||||
|
|
||||||
## SDK
|
|
||||||
待开发
|
### 参数
|
||||||
|
支持证书id和域名两种方式获取证书。
|
||||||
|
|
||||||
|
### 创建新的证书申请
|
||||||
|
参数autoApply=true,将在没有证书时自动触发申请证书,检查逻辑如下:
|
||||||
|
1. 如果证书仓库里面有,且没有过期,就直接返回证书
|
||||||
|
2. 如果没有或者已过期,就会去找流水线,有就触发流水线执行
|
||||||
|
3. 如果没有流水线,就创建一个流水线,触发运行(`注意:需要提前在域名管理中配置好域名校验方式,否则会申请失败`)
|
||||||
|
4. 再次采用相同参数请求接口,如果在申请过程中,就会提示`正在申请中`,可轮循获取状态,直到证书申请成功。
|
||||||
|
|
||||||
|
|
||||||
|
### SDK
|
||||||
|
待开发
|
||||||
|
|
||||||
|
## 客户端工具
|
||||||
|
|
||||||
|
### SSL-Assistant
|
||||||
|
`SSL Assistant` 是一个基于 Go 语言开发的跨平台证书部署管理助手。
|
||||||
|
支持自动扫描主机`Nginx`配置,然后从Certd拉取证书并部署。
|
||||||
|
在不想暴露ssh主机密码情况下,该工具非常好用。
|
||||||
|
|
||||||
|
开源地址: https://github.com/Youngxj/SSL-Assistant
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
# 插件列表
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
|
|
||||||
@@ -0,0 +1,84 @@
|
|||||||
|
# 授权列表
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **主机登录授权** | |
|
||||||
|
| 2.| **阿里云授权** | |
|
||||||
|
| 3.| **阿里云ESA授权** | |
|
||||||
|
| 4.| **华为云授权** | |
|
||||||
|
| 5.| **腾讯云** | |
|
||||||
|
| 6.| **京东云** | |
|
||||||
|
| 7.| **火山引擎** | |
|
||||||
|
| 8.| **七牛云授权** | |
|
||||||
|
| 9.| **百度云授权** | |
|
||||||
|
| 10.| **baota授权** | |
|
||||||
|
| 11.| **天翼云授权** | |
|
||||||
|
| 12.| **51dns授权** | |
|
||||||
|
| 13.| **AcePanel授权** | |
|
||||||
|
| 14.| **SFTP授权** | |
|
||||||
|
| 15.| **阿里云OSS授权** | 包含地域和Bucket |
|
||||||
|
| 16.| **APISIX授权** | |
|
||||||
|
| 17.| **亚马逊云aws授权** | |
|
||||||
|
| 18.| **亚马逊云科技(国区)授权** | |
|
||||||
|
| 19.| **CacheFly** | CacheFly |
|
||||||
|
| 20.| **EAB授权** | ZeroSSL证书申请需要EAB授权 |
|
||||||
|
| 21.| **google cloud** | 谷歌云授权 |
|
||||||
|
| 22.| **cloudflare授权** | |
|
||||||
|
| 23.| **中国移动CND授权** | |
|
||||||
|
| 24.| **授权插件示例** | 这是一个示例授权插件,用于演示如何实现一个授权插件 |
|
||||||
|
| 25.| **dns.la授权** | |
|
||||||
|
| 26.| **多吉云** | |
|
||||||
|
| 27.| **Dokploy授权** | |
|
||||||
|
| 28.| **farcdn授权** | |
|
||||||
|
| 29.| **FlexCDN授权** | |
|
||||||
|
| 30.| **Gcore** | Gcore |
|
||||||
|
| 31.| **Github授权** | |
|
||||||
|
| 32.| **godaddy授权** | |
|
||||||
|
| 33.| **金山云授权** | |
|
||||||
|
| 34.| **FTP授权** | |
|
||||||
|
| 35.| **七牛OSS授权** | |
|
||||||
|
| 36.| **腾讯云COS授权** | 腾讯云对象存储授权,包含地域和存储桶 |
|
||||||
|
| 37.| **s3/minio授权** | S3/minio oss授权 |
|
||||||
|
| 38.| **namesilo授权** | |
|
||||||
|
| 39.| **Next Terminal 授权** | 用于访问 Next Terminal API 的授权配置 |
|
||||||
|
| 40.| **1panel授权** | 账号和密码 |
|
||||||
|
| 41.| **支付宝** | |
|
||||||
|
| 42.| **白山云授权** | |
|
||||||
|
| 43.| **宝塔云WAF授权** | 用于连接和管理宝塔云WAF服务的授权配置 |
|
||||||
|
| 44.| **cdnfly授权** | |
|
||||||
|
| 45.| **k8s授权** | |
|
||||||
|
| 46.| **括彩云cdn授权** | 括彩云CDN,每月免费30G,[注册即领](https://kuocaicdn.com/register?code=8mn536rrzfbf8) |
|
||||||
|
| 47.| **LeCDN授权** | |
|
||||||
|
| 48.| **lucky** | |
|
||||||
|
| 49.| **猫云授权** | |
|
||||||
|
| 50.| **plesk授权** | |
|
||||||
|
| 51.| **长亭雷池授权** | |
|
||||||
|
| 52.| **群晖登录授权** | |
|
||||||
|
| 53.| **uniCloud** | unicloud授权 |
|
||||||
|
| 54.| **微信支付** | |
|
||||||
|
| 55.| **易盾rcdn授权** | 易盾CDN,每月免费30G,[注册即领](https://rhcdn.yiduncdn.com/register?code=8mn536rrzfbf8) |
|
||||||
|
| 56.| **易发云短信** | sms.yfyidc.cn/ |
|
||||||
|
| 57.| **易盾DCDN授权** | https://user.yiduncdn.com |
|
||||||
|
| 58.| **易支付** | |
|
||||||
|
| 59.| **proxmox** | |
|
||||||
|
| 60.| **UCloud授权** | 优刻得授权 |
|
||||||
|
| 61.| **又拍云** | |
|
||||||
|
| 62.| **网宿授权** | |
|
||||||
|
| 63.| **西部数码授权** | |
|
||||||
|
| 64.| **我爱云授权** | 我爱云CDN |
|
||||||
|
| 65.| **新网授权(代理方式)** | |
|
||||||
|
| 66.| **新网授权** | |
|
||||||
|
| 67.| **新网互联授权** | 仅支持代理账号,ip需要加入白名单 |
|
||||||
|
| 68.| **Zenlayer授权** | Zenlayer授权 |
|
||||||
|
| 69.| **GoEdge授权** | |
|
||||||
|
| 70.| **雨云授权** | https://app.rainyun.com/ |
|
||||||
|
|
||||||
|
<style module>
|
||||||
|
table th:first-of-type {
|
||||||
|
width: 65px;
|
||||||
|
}
|
||||||
|
table th:nth-of-type(2) {
|
||||||
|
width: 240px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
@@ -0,0 +1,198 @@
|
|||||||
|
# 任务插件
|
||||||
|
共 `125` 款任务插件
|
||||||
|
## 1. 证书申请
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **证书申请(JS版)** | 免费通配符域名证书申请,支持多个域名打到同一个证书上 |
|
||||||
|
| 2.| **已有证书托管** | 手动上传自定义证书后,自动部署(每次证书有更新,都需要手动上传一次) |
|
||||||
|
| 3.| **获取阿里云订阅证书** | 从阿里云拉取订阅模式的商用证书 |
|
||||||
|
| 4.| **证书申请(Lego)** | 支持海量DNS解析提供商,推荐使用,一样的免费通配符域名证书申请,支持多个域名打到同一个证书上 |
|
||||||
|
## 2. 主机
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **主机-复制到本机** | 【仅管理员使用】实际上是复制证书到docker容器内的某个路径,需要做目录映射到宿主机 |
|
||||||
|
| 2.| **主机-执行远程主机脚本命令** | 可以执行重启nginx等操作让证书生效 |
|
||||||
|
| 3.| **IIS-部署到IIS站点** | |
|
||||||
|
| 4.| **上传证书到对象存储OSS** | 支持阿里云OSS、腾讯云COS、七牛云KODO、S3、MinIO、FTP、SFTP |
|
||||||
|
| 5.| **主机-部署证书到SSH主机** | 上传证书到主机覆盖原来的证书文件,然后自动执行部署脚本命令使证书生效 |
|
||||||
|
| 6.| **ESXi-部署证书到ESXi** | |
|
||||||
|
| 7.| **FTP-上传证书到FTP** | 将证书上传到FTP服务器 |
|
||||||
|
| 8.| **Openwrt-部署证书到Openwrt** | |
|
||||||
|
## 3. CDN
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **APISIX-更新证书** | 自动更新APISIX证书 |
|
||||||
|
| 2.| **CacheFly-部署证书到CacheFly** | 部署证书到 CacheFly |
|
||||||
|
| 3.| **中国移动-部署证书到CDN** | 中国移动自动部署证书到CDN |
|
||||||
|
| 4.| **多吉云-部署到多吉云CDN** | |
|
||||||
|
| 5.| **farcdn-更新证书** | www.farcdn.net |
|
||||||
|
| 6.| **FlexCDN-更新证书** | |
|
||||||
|
| 7.| **Gcore-刷新Gcore证书** | 刷新现有的证书 |
|
||||||
|
| 8.| **Gcore-部署证书到Gcore** | 仅上传 并不会部署到cdn |
|
||||||
|
| 9.| **GoEdge-更新证书** | GoEdge |
|
||||||
|
| 10.| **金山云-更新CDN证书** | 金山云自动更新CDN证书 |
|
||||||
|
| 11.| **白山云-更新证书** | |
|
||||||
|
| 12.| **cdnfly-部署证书到cdnfly** | cdnfly |
|
||||||
|
| 13.| **天翼云-部署证书到CDN** | 部署证书到天翼云CDN和全站加速 |
|
||||||
|
| 14.| **括彩云-部署到括彩云CDN** | 括彩云CDN,每月免费30G,[注册即领](https://kuocaicdn.com/register?code=8mn536rrzfbf8) |
|
||||||
|
| 15.| **LeCDN-更新证书V2** | 支持新版本LeCDN |
|
||||||
|
| 16.| **LeCDN-更新证书** | |
|
||||||
|
| 17.| **Maoyun-更新猫云CDN证书** | |
|
||||||
|
| 18.| **易盾-部署到易盾DCDN** | 主要是防御,http://user.yiduncdn.com/ |
|
||||||
|
| 19.| **易盾-部署到易盾RCDN** | 易盾CDN,每月免费30G,[注册即领](https://rhcdn.yiduncdn.com/register?code=8mn536rrzfbf8) |
|
||||||
|
| 20.| **雨云-更新证书** | app.rainyun.com |
|
||||||
|
| 21.| **又拍云-部署证书到CDN/USS** | 支持又拍云CDN,又拍云云存储USS |
|
||||||
|
| 22.| **网宿-更新证书** | 网宿证书自动更新 |
|
||||||
|
| 23.| **西数-部署到虚拟主机** | 西部数码部署证书到虚拟主机 |
|
||||||
|
| 24.| **我爱云-部署证书到我爱云CDN** | 部署证书到我爱云CDN |
|
||||||
|
| 25.| **Zenlayer-刷新证书** | 刷新Zenlayer CDN证书 |
|
||||||
|
## 4. 面板
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **AcePanel-部署到网站** | 上传证书并部署到指定网站 |
|
||||||
|
| 2.| **AcePanel-面板证书** | 部署AcePanel面板证书 |
|
||||||
|
| 3.| **Dokploy-部署server证书** | 自动更新Dokploy server证书 |
|
||||||
|
| 4.| **飞牛NAS-部署证书** | |
|
||||||
|
| 5.| **NextTerminal-更新证书** | 更新 Next Terminal 证书 |
|
||||||
|
| 6.| **1Panel-部署面板证书** | 更新1Panel的面板证书 |
|
||||||
|
| 7.| **1Panel-更新站点证书** | 更新1Panel的站点证书 |
|
||||||
|
| 8.| **宝塔-删除过期证书** | 删除证书夹中过期证书 |
|
||||||
|
| 9.| **宝塔-WAF证书部署** | 部署宝塔云WAF/aaWAF |
|
||||||
|
| 10.| **宝塔-面板证书部署** | 部署宝塔面板本身的ssl证书 |
|
||||||
|
| 11.| **宝塔win-网站证书部署** | 部署到Windows版宝塔管理的站点的ssl证书 |
|
||||||
|
| 12.| **宝塔-网站证书部署** | 部署宝塔管理的站点的ssl证书,目前支持宝塔网站站点、docker站点等。本插件也支持aaPanel。 |
|
||||||
|
| 13.| **K8S-Apply自定义yaml** | apply自定义yaml到k8s |
|
||||||
|
| 14.| **K8S-Ingress 证书部署** | 部署证书到k8s的Ingress |
|
||||||
|
| 15.| **K8S-部署证书到Secret** | 部署证书到k8s的secret |
|
||||||
|
| 16.| **lucky-更新Lucky证书** | |
|
||||||
|
| 17.| **Plesk-部署Plesk网站证书** | |
|
||||||
|
| 18.| **Plesk-更新证书** | 不会创建新证书记录,直接更新旧的证书 |
|
||||||
|
| 19.| **雷池-更新证书(支持控制台和防护应用)** | 更新长亭雷池WAF的证书,支持更新控制台和防护应用的证书。 |
|
||||||
|
| 20.| **群晖-部署证书到群晖面板** | Synology,支持6.x以上版本 |
|
||||||
|
| 21.| **群晖-刷新OTP登录有效期** | 群晖登录状态可能30天失效,需要在失效之前登录一次,刷新有效期,您可以将其放在“部署到群晖面板”任务之后 |
|
||||||
|
| 22.| **uniCloud-部署到服务空间** | 部署到服务空间 |
|
||||||
|
| 23.| **Proxmox-上传证书到Proxmox** | |
|
||||||
|
| 24.| **威联通-部署证书到威联通** | 部署证书到qnap |
|
||||||
|
## 5. 阿里云
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **阿里云-部署到Ack** | 部署到阿里云Ack集群Ingress等通过Secret管理证书的应用 |
|
||||||
|
| 2.| **阿里云-部署至ALB(应用负载均衡)** | ALB,更新监听器的默认证书 |
|
||||||
|
| 3.| **阿里云-部署至任意云资源** | 【不建议使用】需要消耗阿里云自动部署次数,支持SLB、LIVE、webHosting、VOD、CR、DCDN、DDoS、CDN、ALB、APIGateway、FC、GA、MSE、NLB、OSS、SAE、WAF等云产品 |
|
||||||
|
| 4.| **阿里云-部署至云原生API网关/AI网关** | 自动部署域名证书至云原生API网关、AI网关 |
|
||||||
|
| 5.| **阿里云-部署证书至API网关** | 自动部署域名证书至阿里云API网关(APIGateway) |
|
||||||
|
| 6.| **阿里云-部署证书至CDN** | 自动部署域名证书至阿里云CDN |
|
||||||
|
| 7.| **阿里云-部署证书至DCDN** | 依赖证书申请前置任务,自动部署域名证书至阿里云DCDN |
|
||||||
|
| 8.| **阿里云-部署至ESA** | 部署证书到阿里云ESA(边缘安全加速),自动删除过期证书 |
|
||||||
|
| 9.| **阿里云-部署至阿里云FC(3.0)** | 部署证书到阿里云函数计算(FC3.0) |
|
||||||
|
| 10.| **阿里云-部署至GA** | 部署证书到阿里云GA(全球加速),支持更新默认证书和扩展证书 |
|
||||||
|
| 11.| **阿里云-部署至NLB(网络负载均衡)** | NLB,网络负载均衡,更新监听器的默认证书 |
|
||||||
|
| 12.| **阿里云-部署证书至OSS** | 部署域名证书至阿里云OSS自定义域名,不是上传到阿里云oss |
|
||||||
|
| 13.| **阿里云-部署至CLB(传统负载均衡)** | 部署证书到阿里云CLB(传统负载均衡) |
|
||||||
|
| 14.| **阿里云-部署至VOD** | 部署证书到阿里云视频点播(vod) |
|
||||||
|
| 15.| **阿里云-部署至阿里云WAF** | 部署证书到阿里云WAF |
|
||||||
|
| 16.| **阿里云-上传证书到CAS** | 上传证书到阿里云证书管理服务(CAS),如果不想在阿里云上同一份证书上传多次,可以把此任务作为前置任务,其他阿里云任务证书那一项选择此任务的输出 |
|
||||||
|
## 6. 华为云
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **华为云-部署证书至CDN** | |
|
||||||
|
| 2.| **华为云-部署证书至ELB负载均衡** | |
|
||||||
|
| 3.| **华为云-部署证书至OBS** | |
|
||||||
|
| 4.| **华为云-上传证书至CCM** | 上传证书到华为云云证书管理(CCM) |
|
||||||
|
## 7. 腾讯云
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **腾讯云-删除即将过期证书** | 仅删除未使用的证书 |
|
||||||
|
| 2.| **腾讯云-部署证书到任意云资源** | 支持负载均衡、CDN、DDoS、直播、点播、Web应用防火墙、API网关、TEO、容器服务、对象存储、轻应用服务器、云原生微服务、云开发 |
|
||||||
|
| 3.| **腾讯云-部署到CDN(废弃)** | 已废弃,请使用v2版 |
|
||||||
|
| 4.| **腾讯云-部署到CDN-v2** | 推荐使用 |
|
||||||
|
| 5.| **腾讯云-部署到CLB** | 暂时只支持单向认证证书,暂时只支持通用负载均衡 |
|
||||||
|
| 6.| **腾讯云-部署证书到COS** | 部署到腾讯云COS源站域名证书【注意:很不稳定,需要重试很多次偶尔才能成功一次】 |
|
||||||
|
| 7.| **腾讯云-部署到腾讯云EO** | 腾讯云边缘安全加速平台EdgeOne(EO) |
|
||||||
|
| 8.| **腾讯云-部署到腾讯云直播** | https://console.cloud.tencent.com/live/ |
|
||||||
|
| 9.| **腾讯云-部署到TKE** | 修改TKE集群密钥配置,支持Opaque和TLS证书类型。注意: 1. serverless集群请使用K8S部署插件; 2. Opaque类型需要【上传到腾讯云】作为前置任务; 3. ApiServer需要开通公网访问(或者certd可访问),实际上底层仍然是通过KubeClient进行部署 |
|
||||||
|
| 10.| **腾讯云-更新证书(Id不变)** | 根据证书id一键更新腾讯云证书并自动部署(Id不变),注意:当前仅支持CLB,其他需要等腾讯接口完善 |
|
||||||
|
| 11.| **腾讯云-实例开关机** | 腾讯云实例开关机 |
|
||||||
|
| 12.| **腾讯云-上传证书到腾讯云** | 上传成功后输出:tencentCertId |
|
||||||
|
## 8. 火山引擎
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **火山引擎-部署证书至ALB** | 部署至火山引擎应用负载均衡 |
|
||||||
|
| 2.| **火山引擎-部署证书至CDN** | 支持网页,文件下载,音视频点播 |
|
||||||
|
| 3.| **火山引擎-部署证书至CLB** | 部署至火山引擎负载均衡 |
|
||||||
|
| 4.| **火山引擎-部署证书至DCDN** | 部署至火山引擎全站加速 |
|
||||||
|
| 5.| **火山引擎-部署证书至Live** | 部署至火山引擎视频直播 |
|
||||||
|
| 6.| **火山引擎-部署证书至VOD** | 部署至火山引擎视频点播(暂不可用) |
|
||||||
|
| 7.| **火山引擎-上传证书至证书中心** | 上传证书至火山引擎证书中心 |
|
||||||
|
## 9. 京东云
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **京东云-部署证书至CDN** | 京东云内容分发网络 |
|
||||||
|
| 2.| **京东云-更新已有证书** | 更新SSL数字证书中的证书 |
|
||||||
|
| 3.| **京东云-上传新证书** | 上传证书到SSL数字证书中心 |
|
||||||
|
## 10. UCloud
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **UCloud-部署到CDN** | 将证书部署到UCloud CDN |
|
||||||
|
| 2.| **UCloud-部署到负载均衡** | 将证书部署到UCloud负载均衡(ULB/ALB/CLB) |
|
||||||
|
| 3.| **UCloud-部署到对象存储(US3)** | 将证书部署到UCloud对象存储(US3) |
|
||||||
|
| 4.| **UCloud-部署到WAF** | 将证书部署到UCloud WAF |
|
||||||
|
| 5.| **UCloud-上传到USSL** | 将证书上传到UCloud USSL |
|
||||||
|
## 11. 百度云
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **百度云-部署证书到负载均衡** | 部署到百度云负载均衡,包括BLB、APPBLB |
|
||||||
|
| 2.| **百度云-部署证书到CDN** | 部署到百度云CDN |
|
||||||
|
| 3.| **百度云-上传到证书托管** | 上传证书到百度云证书托管中心 |
|
||||||
|
## 12. 七牛云
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **七牛云-部署证书至CDN/DCDN** | 自动部署域名证书至七牛云CDN、DCDN |
|
||||||
|
| 2.| **七牛云-部署证书至OSS** | 自动部署域名证书至七牛云KODO,注意是自定义源站域名,不是CDN域名 |
|
||||||
|
| 3.| **七牛云-上传证书到七牛云** | 上传到七牛云 |
|
||||||
|
## 13. 亚马逊云
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **AWS-部署证书到CloudFront** | 部署证书到 AWS CloudFront |
|
||||||
|
| 2.| **AWS-上传证书到ACM** | 上传证书 AWS ACM |
|
||||||
|
| 3.| **AWS(国区)-部署证书到CloudFront** | 部署证书到 AWS CloudFront |
|
||||||
|
## 14. 其他
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **Demo-测试插件** | |
|
||||||
|
| 2.| **Github-检查Release版本** | 检查最新Release版本并推送消息 |
|
||||||
|
| 3.| **邮件发送证书** | 通过邮件发送证书 |
|
||||||
|
| 4.| **等待** | 等待一段时间 |
|
||||||
|
| 5.| **webhook方式部署证书** | 调用webhook部署证书 |
|
||||||
|
## 15. 管理
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **数据库备份** | 【仅管理员可用】仅支持备份SQLite数据库 |
|
||||||
|
| 2.| **重启 Certd** | 【仅管理员可用】 重启 certd的https服务,用于更新 Certd 的 ssl 证书 |
|
||||||
|
| 3.| **部署证书到Certd本身** | 【仅管理员可用】 部署证书到 certd的https服务,用于更新 Certd 的 ssl 证书,建议将此任务放在流水线的最后一步 |
|
||||||
|
| 4.| **自定义js脚本** | 【仅管理员】运行自定义js脚本执行 |
|
||||||
|
|
||||||
|
<style module>
|
||||||
|
table th:first-of-type {
|
||||||
|
width: 65px;
|
||||||
|
}
|
||||||
|
table th:nth-of-type(2) {
|
||||||
|
width: 240px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
# DNS提供商
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **阿里ESA** | 阿里ESA DNS解析 |
|
||||||
|
| 2.| **阿里云** | 阿里云DNS解析提供商 |
|
||||||
|
| 3.| **AWS Route53** | AWS Route53 DNS解析提供商 |
|
||||||
|
| 4.| **火山引擎** | 火山引擎DNS解析提供商 |
|
||||||
|
| 5.| **京东云** | 京东云DNS解析提供商 |
|
||||||
|
| 6.| **新网(代理方式)** | 新网域名解析(代理方式) |
|
||||||
|
| 7.| **新网** | 新网域名解析 |
|
||||||
|
| 8.| **cloudflare** | cloudflare dns provider |
|
||||||
|
| 9.| **dns.la** | dns.la |
|
||||||
|
| 10.| **godaddy** | GoDaddy |
|
||||||
|
| 11.| **华为云** | 华为云DNS解析提供商 |
|
||||||
|
| 12.| **namesilo** | namesilo dns provider |
|
||||||
|
| 13.| **雨云** | 雨云DNS解析提供商 |
|
||||||
|
| 14.| **腾讯云** | 腾讯云域名DNS解析提供者 |
|
||||||
|
| 15.| **腾讯云EO DNS** | 腾讯云EO DNS解析提供者 |
|
||||||
|
| 16.| **西部数码** | west dns provider |
|
||||||
|
| 17.| **Dns提供商Demo** | dns provider示例 |
|
||||||
|
| 18.| **51dns** | 51DNS |
|
||||||
|
| 19.| **新网互联** | 新网互联 |
|
||||||
|
|
||||||
|
<style module>
|
||||||
|
table th:first-of-type {
|
||||||
|
width: 65px;
|
||||||
|
}
|
||||||
|
table th:nth-of-type(2) {
|
||||||
|
width: 240px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
# 通知插件
|
||||||
|
|
||||||
|
| 序号 | 名称 | 说明 |
|
||||||
|
|-----|-----|-----|
|
||||||
|
| 1.| **电子邮件** | 电子邮件通知 |
|
||||||
|
| 2.| **自定义webhook** | 根据模版自定义http请求 |
|
||||||
|
| 3.| **AnPush** | https://anpush.com |
|
||||||
|
| 4.| **Bark 通知** | Bark 推送通知插件 |
|
||||||
|
| 5.| **钉钉通知** | 钉钉群聊通知 |
|
||||||
|
| 6.| **Discord 通知** | Discord 机器人通知 |
|
||||||
|
| 7.| **飞书通知** | 飞书群聊webhook通知 |
|
||||||
|
| 8.| **爱语飞飞微信通知(iyuu)** | https://iyuu.cn/ |
|
||||||
|
| 9.| **MeoW通知** | https://api.chuckfang.com/ |
|
||||||
|
| 10.| **OneBot V11 通知** | 通过动态拼接URL发送 OneBot V11 协议消息 |
|
||||||
|
| 11.| **企业微信通知** | 企业微信群聊机器人通知 |
|
||||||
|
| 12.| **Server酱ᵀ** | https://sct.ftqq.com/ |
|
||||||
|
| 13.| **Server酱³** | https://doc.sc3.ft07.com/serverchan3 |
|
||||||
|
| 14.| **Slack通知** | Slack消息推送通知 |
|
||||||
|
| 15.| **Telegram通知** | Telegram Bot推送通知 |
|
||||||
|
| 16.| **VoceChat通知** | https://voce.chat |
|
||||||
|
|
||||||
|
<style module>
|
||||||
|
table th:first-of-type {
|
||||||
|
width: 65px;
|
||||||
|
}
|
||||||
|
table th:nth-of-type(2) {
|
||||||
|
width: 240px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
After Width: | Height: | Size: 82 KiB |
@@ -0,0 +1,62 @@
|
|||||||
|
# 常见问题
|
||||||
|
|
||||||
|
|
||||||
|
## 1. 是否支持IP证书
|
||||||
|
|
||||||
|
因为ACME协议不支持IP证书,所以certd目前也不支持IP证书
|
||||||
|
|
||||||
|
|
||||||
|
## 2. 建议设置多长时间运行一次流水线
|
||||||
|
建议每天运行一次,检查证书过期时间
|
||||||
|
当证书没过期时,自动跳过部署
|
||||||
|
当证书到期前35天(创建流水线时可以修改),将会自动重新申请证书,自动部署
|
||||||
|
|
||||||
|
|
||||||
|
## 3. too many certificates 错误
|
||||||
|
当出现如下报错时,说明相同的域名短时间内申请超过5次
|
||||||
|
解决方案:可以加多一个子域名,重新执行就可以规避次错误
|
||||||
|
```
|
||||||
|
"detail": too many certificates (5) already issued for this exact set of idantifiers in the last 168hm0s
|
||||||
|
```
|
||||||
|
|
||||||
|
## 4. ssl.com报错 CAA record does not include ssl.com which is required to issue the certificate
|
||||||
|
ssl.com申请证书要求必须设置CAA记录,表示允许ssl.com为该域名颁发证书
|
||||||
|
请按如下格式添加CAA记录
|
||||||
|
|
||||||
|
| 示例 | 类型 | 域名前缀 | flag | tag | 值 |
|
||||||
|
|-------|-----| -- |-----------|--------|----------------------|
|
||||||
|
| 顶级域名 | CAA | @ | 0 | issue | "ssl.com" (注意有双引号) |
|
||||||
|
| 一级泛域名 | CAA | * | 0 | issue/issuewild | "ssl.com" |
|
||||||
|
| 固定子域名 | CAA | sub | 0 | issue |"ssl.com" |
|
||||||
|
|
||||||
|
## 5. address family not supported
|
||||||
|
启动时出现此错误,是由于您的服务器不支持绑定ipv6地址
|
||||||
|
|
||||||
|
请配置环境变量 certd_koa_hostname=0.0.0.0
|
||||||
|
|
||||||
|
在docker-compose.yml中添加如下配置
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
service:
|
||||||
|
certd:
|
||||||
|
environment:
|
||||||
|
certd_koa_hostname: 0.0.0.0
|
||||||
|
```
|
||||||
|
|
||||||
|
## 6. DNS记录问题
|
||||||
|
|
||||||
|
1. DNS 不要设置CAA记录,删除即可
|
||||||
|
|
||||||
|
2. DNSSEC相关报错,DNSSEC管理中删除即可
|
||||||
|
|
||||||
|
3. DNS 有其他平台申请过的_acme-challenge记录,删除即可
|
||||||
|
|
||||||
|
|
||||||
|
## 7. DNS problem: NXDOMAIN looking up TXT for _acme-challenge.xxx
|
||||||
|
`
|
||||||
|
DNS problem: NXDOMAIN looking up TXT for _acme-challenge.xxxxx - check that a DNS record exists for this domain
|
||||||
|
`
|
||||||
|
证书颁发机构向域名ns查询TXT验证记录失败,有以下几种可能
|
||||||
|
1、域名的ns服务器修改成别的了,但申请证书时的DNS提供商选择错误(检查确认,配置正确的DNS提供商)
|
||||||
|
2、证书颁发机构与ns域名服务器之间访问不通,无法查询到TXT记录(尝试更换证书颁发机构)
|
||||||
|
3、ns服务商解析值生效慢(尝试修改证书申请任务里面的等待生效时长600-1000s)
|
||||||
@@ -7,10 +7,16 @@
|
|||||||
|
|
||||||
https://certd.handsfree.work/
|
https://certd.handsfree.work/
|
||||||
|
|
||||||
> 注意数据将不定期清理,不定期停止定时任务,生产使用请自行部署
|
注册 -> 创建证书流水线 -> 添加部署任务 -> 测试运行
|
||||||
|
|
||||||
|
> 注意demo的数据将不定期清理,生产使用请自行部署
|
||||||
> 包含敏感信息,务必自己本地部署进行生产使用
|
> 包含敏感信息,务必自己本地部署进行生产使用
|
||||||
|
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## 二、私有化部署
|
## 二、私有化部署
|
||||||
|
|
||||||
由于证书、授权信息等属于高度敏感数据,请务必私有化部署,保障数据安全
|
由于证书、授权信息等属于高度敏感数据,请务必私有化部署,保障数据安全
|
||||||
|
|||||||
@@ -15,3 +15,4 @@
|
|||||||
## 2. 图文教程链接
|
## 2. 图文教程链接
|
||||||
如果不方便登录系统,您还可以直接查看 [图文教程](https://gitee.com/certd/certd/blob/v2/step.md)
|
如果不方便登录系统,您还可以直接查看 [图文教程](https://gitee.com/certd/certd/blob/v2/step.md)
|
||||||
|
|
||||||
|

|
||||||
|
Before Width: | Height: | Size: 47 KiB After Width: | Height: | Size: 47 KiB |
@@ -5,8 +5,10 @@
|
|||||||
|
|
||||||
配置环境变量
|
配置环境变量
|
||||||
```shell
|
```shell
|
||||||
ALIYUN_CLIENT_CONNECT_TIMEOUT=10000 # 连接超时,单位毫秒
|
# docker-compose.yaml
|
||||||
ALIYUN_CLIENT_READ_TIMEOUT=10000 #读取数据超时,单位毫秒
|
environment:
|
||||||
|
- ALIYUN_CLIENT_CONNECT_TIMEOUT=16000 # 连接超时,单位毫秒
|
||||||
|
- ALIYUN_CLIENT_READ_TIMEOUT=16000 #读取数据超时,单位毫秒
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
# 数据库自动备份
|
# 数据库备份
|
||||||
|
* 两种备份方法: 1、手动备份 2、自动备份
|
||||||
|
* 本文仅限sqlite数据库。
|
||||||
## 一、手动备份
|
## 一、手动备份
|
||||||
数据库文件根据不同的部署方式保存的位置不一样,您可以手动复制出来进行备份
|
数据库文件根据不同的部署方式保存的位置不一样,您可以手动复制出来进行备份
|
||||||
|
|
||||||
|
|||||||
|
After Width: | Height: | Size: 22 KiB |
|
After Width: | Height: | Size: 30 KiB |
|
After Width: | Height: | Size: 36 KiB |
|
After Width: | Height: | Size: 15 KiB |
|
After Width: | Height: | Size: 7.2 KiB |
|
After Width: | Height: | Size: 27 KiB |
|
After Width: | Height: | Size: 58 KiB |
|
After Width: | Height: | Size: 23 KiB |
@@ -0,0 +1,98 @@
|
|||||||
|
# 宝塔IP白名单与动态IP问题
|
||||||
|
调用宝塔接口需要添加IP白名单,但当certd部署在动态IP环境下时,IP白名单就不好添加
|
||||||
|
本章节提供两种解决方案:
|
||||||
|
1. 小范围网段放开(简单)
|
||||||
|
2. nginx代理
|
||||||
|
|
||||||
|
## 一、放开小范围网段
|
||||||
|
|
||||||
|
家庭网络IP虽然会变动,但是只会在小范围变的。
|
||||||
|
|
||||||
|
你可以分析规律,将变动的部分,设置成网段即可
|
||||||
|
|
||||||
|
> 比如出现过: 100.25.1.5 , 100.25.1.8
|
||||||
|
>
|
||||||
|
> 那么你可以配置 100.25.1.1-100.25.1.255
|
||||||
|
|
||||||
|
|
||||||
|
> 如果出现过: 100.25.1.5 , 100.25.4.8
|
||||||
|
>
|
||||||
|
> 可以尝试配置 100.25.*.*
|
||||||
|
|
||||||
|
## 二、nginx代理方案
|
||||||
|
|
||||||
|
通过在宝塔中配置一个nginx反向代理,代理宝塔自己的地址
|
||||||
|
|
||||||
|
然后在nginx中配置放开certd需要的接口,缩小影响范围
|
||||||
|
|
||||||
|
让nginx来充当防火墙
|
||||||
|
|
||||||
|
架构图如下:
|
||||||
|
```
|
||||||
|
只要将127.0.0.1加入白名单即可
|
||||||
|
↓
|
||||||
|
certd --------> nginx -------> 宝塔
|
||||||
|
↑
|
||||||
|
拦截除更新证书之外的地址
|
||||||
|
```
|
||||||
|
|
||||||
|
### 1. 添加nginx反向代理
|
||||||
|

|
||||||
|
|
||||||
|
### 2. 域名和代理目标
|
||||||
|

|
||||||
|
|
||||||
|
### 3. 设置放开哪些接口
|
||||||
|

|
||||||
|

|
||||||
|
将如下脚本填入上方文本域中,保存
|
||||||
|
```nginx configuration
|
||||||
|
set $allow_access false;
|
||||||
|
|
||||||
|
# 检查请求的URI是否在白名单中
|
||||||
|
if ($request_uri ~* "^/(site\?action=get_site_types)") {
|
||||||
|
# 允许测试
|
||||||
|
set $allow_access true;
|
||||||
|
}
|
||||||
|
if ($request_uri ~* "^/(config\?action=SavePanelSSL)") {
|
||||||
|
# 允许部署到宝塔面板本身证书
|
||||||
|
set $allow_access true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($request_uri ~* "^/(mod/docker/com/set_ssl|site\?action=SetSSL|ssl\?action=GetSiteDomain|mod/docker/com/get_site_list)") {
|
||||||
|
# 允许部署宝塔网站证书
|
||||||
|
set $allow_access true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($request_uri ~* "^/(ssl?action=remove_cloud_cert|ssl\?action=get_cert_list)") {
|
||||||
|
# 允许删除宝塔过期证书
|
||||||
|
set $allow_access true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($request_uri ~* "^/(datalist/get_data_list|site/set_site_ssl)") {
|
||||||
|
set $allow_access true;
|
||||||
|
}
|
||||||
|
|
||||||
|
# 如果不在白名单,返回403禁止访问
|
||||||
|
if ($allow_access = false) {
|
||||||
|
return 405;
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### 4. 接口IP白名单添加127.0.0.1
|
||||||
|

|
||||||
|
|
||||||
|
### 5. certd中宝塔授权配置改成新的这个域名地址
|
||||||
|
|
||||||
|

|
||||||
|
点击测试检查是否ok ,到这里就可以正常部署证书了
|
||||||
|
|
||||||
|
### 6. 安全加强(将请求地址改成https)
|
||||||
|
在宝塔中配置证书部署任务,选择刚才新建的这个网站,给他部署证书
|
||||||
|
勾选强制https
|
||||||
|

|
||||||
|
更换443端口【可选】
|
||||||
|

|
||||||
|
禁止http访问
|
||||||