mirror of
				https://github.com/containous/traefik.git
				synced 2025-10-25 03:33:20 +03:00 
			
		
		
		
	Compare commits
	
		
			1113 Commits
		
	
	
		
			v2.2.2
			...
			v2.11.0-rc
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 9adf0fb638 | ||
|  | 56e2110dc5 | ||
|  | 5be13802dc | ||
|  | 7345afd8b6 | ||
|  | a84d5c0ef1 | ||
|  | 2a9471d278 | ||
|  | 0042562678 | ||
|  | 74ab88d47e | ||
|  | 6df9578ace | ||
|  | cd7d324295 | ||
|  | 0e92b02474 | ||
|  | 9662cdca64 | ||
|  | 3dfaa3d5fa | ||
|  | 60123a8f3f | ||
|  | 2a7b2ef772 | ||
|  | 0a79643001 | ||
|  | e77a66c2ac | ||
|  | dae0491b61 | ||
|  | f4ddf25e41 | ||
|  | 789046f162 | ||
|  | 186e3e1541 | ||
|  | 088fe3c270 | ||
|  | 553ef94047 | ||
|  | 12e50e20e6 | ||
|  | cd326654a7 | ||
|  | 3de29433f8 | ||
|  | 84516f962d | ||
|  | f92b03a44d | ||
|  | 085b70c94e | ||
|  | 0e66ed87f8 | ||
|  | 679975beec | ||
|  | 8faed97e74 | ||
|  | 0b4c582088 | ||
|  | 6a34f238ce | ||
|  | 4b2c763cf3 | ||
|  | d03d8d53fd | ||
|  | e95fde5652 | ||
|  | ab7993428d | ||
|  | b966215e6c | ||
|  | b786f58f80 | ||
|  | 173154cf59 | ||
|  | c3880a69ca | ||
|  | 4d63eb30f9 | ||
|  | dbc679dc30 | ||
|  | fc7f732029 | ||
|  | ba912e1a93 | ||
|  | 3216c8ab10 | ||
|  | 561c580701 | ||
|  | 3fd5c747a2 | ||
|  | b6b6cef3da | ||
|  | d651d1e7cf | ||
|  | 6f22b9e0a7 | ||
|  | f29325c679 | ||
|  | 57780d8004 | ||
|  | 1d85515aac | ||
|  | d948784d38 | ||
|  | 1ddb0afb24 | ||
|  | f518676238 | ||
|  | db3e8a7f5a | ||
|  | 0bd367ebbd | ||
|  | 3f93e9ea71 | ||
|  | 1709f3854c | ||
|  | ebde81e91c | ||
|  | 47faae25d7 | ||
|  | 7792d197e6 | ||
|  | deb4235028 | ||
|  | bed6069e82 | ||
|  | e29da5ad65 | ||
|  | 48de3b0230 | ||
|  | 00048a8351 | ||
|  | 2df5defd36 | ||
|  | aaa763b7af | ||
|  | 8a68ece2cc | ||
|  | 0a861716d4 | ||
|  | 7741c68eaa | ||
|  | 18077ff69a | ||
|  | fa555d0d29 | ||
|  | 0e5898b2f8 | ||
|  | aae76408e2 | ||
|  | 9cc9ed6a0c | ||
|  | e62fe64ec9 | ||
|  | 6885e410f0 | ||
|  | 68ed875966 | ||
|  | d1bdeb3a92 | ||
|  | 878e7de56a | ||
|  | 27353d0740 | ||
|  | c5f23493ab | ||
|  | db515195f0 | ||
|  | 9aa57f362b | ||
|  | 6977b68b72 | ||
|  | 8d8717d421 | ||
|  | 021f37ff71 | ||
|  | 511762cbf3 | ||
|  | 1522afe2ec | ||
|  | 9c73c4c584 | ||
|  | 8f206ce319 | ||
|  | 65c59c9a09 | ||
|  | e044e2b765 | ||
|  | 7805c683e3 | ||
|  | e38c0c3969 | ||
|  | c0e03ae17d | ||
|  | 9060522414 | ||
|  | bb4eb32b1c | ||
|  | 30f991effa | ||
|  | 6082b22922 | ||
|  | 5635687a3e | ||
|  | a3f1009170 | ||
|  | c4bea197ab | ||
|  | e8878fe6ac | ||
|  | f344239bef | ||
|  | 4ed3964b35 | ||
|  | 11966c2098 | ||
|  | 0d1bb72306 | ||
|  | 4c9765b52d | ||
|  | 5f514b0d16 | ||
|  | 01f346f239 | ||
|  | be1b1a6489 | ||
|  | ae65d5ff78 | ||
|  | 7fc07c31a0 | ||
|  | f2eda3aa6d | ||
|  | ac9d88e5a2 | ||
|  | 77509b0913 | ||
|  | be702c2b61 | ||
|  | 54f6144ef2 | ||
|  | a020ab640d | ||
|  | f7be1e97df | ||
|  | 48a2c8e41c | ||
|  | 358f47443e | ||
|  | 3b9e155807 | ||
|  | 2083e4bc16 | ||
|  | c823879097 | ||
|  | 4bc2305ed3 | ||
|  | 99d779a546 | ||
|  | 6e460cd652 | ||
|  | 7af9d16208 | ||
|  | b3f162a8a6 | ||
|  | 4aa3496092 | ||
|  | bbe6a5c07b | ||
|  | 21c455cf20 | ||
|  | 667b2a4078 | ||
|  | 4ae07d91a4 | ||
|  | 7bdf13ebdc | ||
|  | 807feef176 | ||
|  | 7202038649 | ||
|  | dd710dbeb7 | ||
|  | 2e6e5cbd03 | ||
|  | ab36ea7844 | ||
|  | cfef9d9df2 | ||
|  | 9ce69fbdef | ||
|  | 1a6dfe1f6b | ||
|  | e053eb6f17 | ||
|  | 780936eff9 | ||
|  | 0503253cfe | ||
|  | 39331e41a8 | ||
|  | 38f5024ed0 | ||
|  | 479878503d | ||
|  | 6f6c1f7fec | ||
|  | e50bf21a84 | ||
|  | d66875f903 | ||
|  | b995a11d63 | ||
|  | f01a668d53 | ||
|  | e82976e001 | ||
|  | c9e9e8dee2 | ||
|  | 0861c47e54 | ||
|  | e1e86763e3 | ||
|  | b9a175f5c2 | ||
|  | 943811fad6 | ||
|  | 29b8b6911e | ||
|  | e7baf44a2e | ||
|  | 74ef79ea23 | ||
|  | da93dab828 | ||
|  | d97d3a6726 | ||
|  | a8df674dcf | ||
|  | abd569701f | ||
|  | 7e3fe48b80 | ||
|  | 778188ed34 | ||
|  | 88603810a8 | ||
|  | c7647b4938 | ||
|  | af71443b61 | ||
|  | 18d66d7432 | ||
|  | 7c72780820 | ||
|  | 68e8eb2435 | ||
|  | 81a5b1b4c8 | ||
|  | 52e6ce95cf | ||
|  | d547718fdd | ||
|  | af4e74c39d | ||
|  | f6b7940b76 | ||
|  | f1b91a119d | ||
|  | 35d8281f4d | ||
|  | 00de5c711a | ||
|  | b935c80dbd | ||
|  | 22c6630412 | ||
|  | b2c4221429 | ||
|  | 97de552e06 | ||
|  | 454f552691 | ||
|  | 7258048403 | ||
|  | 15f7472091 | ||
|  | a041a6b198 | ||
|  | 7582da9650 | ||
|  | ccbbd0d766 | ||
|  | 93212125e3 | ||
|  | be3b798dd6 | ||
|  | 8128d6ca26 | ||
|  | 14ab1514dc | ||
|  | 40242294d8 | ||
|  | 996eccf5b7 | ||
|  | e9de061b84 | ||
|  | 0ca1c8aac3 | ||
|  | 2c550c284d | ||
|  | 87815586be | ||
|  | 09d6383621 | ||
|  | 39b0077725 | ||
|  | e2a9caf760 | ||
|  | bc79796c38 | ||
|  | b1db81d8ac | ||
|  | ae7db879d9 | ||
|  | dd34905ea9 | ||
|  | 3812e6f3cb | ||
|  | 627175694d | ||
|  | 82cf6c9577 | ||
|  | 63a1186d3e | ||
|  | f75f636e27 | ||
|  | 615dc7fd35 | ||
|  | 52b6b057f0 | ||
|  | 7b3faef4b3 | ||
|  | 7758880f3f | ||
|  | d04903edb2 | ||
|  | a63d5c95a8 | ||
|  | bb66950197 | ||
|  | c4cc30ccc6 | ||
|  | 9cd54baca4 | ||
|  | 7ac687a0a9 | ||
|  | 83ae1021f6 | ||
|  | 67e3bc6380 | ||
|  | 4b291b2cf8 | ||
|  | 89870ad539 | ||
|  | 5bc03af75f | ||
|  | 30ec5c58fe | ||
|  | a4b447256b | ||
|  | 1c9a7b8c61 | ||
|  | d06573de6c | ||
|  | 6c2c561d8f | ||
|  | e5309a4601 | ||
|  | e9f98fb6eb | ||
|  | b351266b2d | ||
|  | fd95560c66 | ||
|  | 788f8fa951 | ||
|  | 89dc466b23 | ||
|  | ab8d7d2e78 | ||
|  | a002ccfce3 | ||
|  | 693d5da1b9 | ||
|  | 8ddc37d528 | ||
|  | 0cb2652f51 | ||
|  | fe8e7ab5b8 | ||
|  | d531963f95 | ||
|  | d578ed7327 | ||
|  | 10528c973a | ||
|  | 56a1ed4220 | ||
|  | 37b6edb28c | ||
|  | 44a2b85dba | ||
|  | 77c8d60092 | ||
|  | b33c8cec0b | ||
|  | 52df1d63fe | ||
|  | c84378d649 | ||
|  | 12dccc4fdd | ||
|  | 32e44816c9 | ||
|  | 23c74c9f2e | ||
|  | 9a82d96e68 | ||
|  | d9589878fb | ||
|  | 703de5331b | ||
|  | d3e4d56a0d | ||
|  | adf82d72ae | ||
|  | 25027d6df8 | ||
|  | e56dfeb7d5 | ||
|  | 5ca7fff7f6 | ||
|  | dfa1f3fc00 | ||
|  | b26c45af2b | ||
|  | 626da4c0ae | ||
|  | 9c02612f65 | ||
|  | b3f4f6bb21 | ||
|  | 2cac58d9c0 | ||
|  | a553085689 | ||
|  | 6dd63e1702 | ||
|  | 868ab7a5c8 | ||
|  | 23c26d64ee | ||
|  | 63f9ec9c38 | ||
|  | 40db06204b | ||
|  | 4755bb2f33 | ||
|  | 45453b20fa | ||
|  | 40d2421db9 | ||
|  | af749f1864 | ||
|  | 1576ad85b8 | ||
|  | 2a2ea759d1 | ||
|  | b4ee7bdcbe | ||
|  | 146991efda | ||
|  | ab94bbaece | ||
|  | 5a706296f2 | ||
|  | 5b3354b8ce | ||
|  | 7751fb24eb | ||
|  | f85f3b68aa | ||
|  | b361608693 | ||
|  | cdda9a18ab | ||
|  | 3686f95832 | ||
|  | 2cb011f595 | ||
|  | b7199a7a9b | ||
|  | 14eb56cf30 | ||
|  | ff2911d070 | ||
|  | f07fcd3d54 | ||
|  | 0e4b4c1a31 | ||
|  | 154d8470ab | ||
|  | c9520480c2 | ||
|  | 05c3486347 | ||
|  | 0231db05b4 | ||
|  | 4dc379c601 | ||
|  | 8f6463ba7a | ||
|  | aff334ffb4 | ||
|  | 28da781194 | ||
|  | 51a02caea3 | ||
|  | 839bc7b3a8 | ||
|  | 9c79fafeeb | ||
|  | c51e590591 | ||
|  | 9c4b336f3b | ||
|  | aa8fda5eae | ||
|  | 8b22101236 | ||
|  | 3c1d5e0393 | ||
|  | 03598d395b | ||
|  | 9d61cb64a2 | ||
|  | ba3f5b318c | ||
|  | 62e17c659e | ||
|  | 41748c3ae4 | ||
|  | 65a317010b | ||
|  | a887794313 | ||
|  | 77e1ce2877 | ||
|  | 470a4f6e5f | ||
|  | 94141233f0 | ||
|  | 467c8b31c3 | ||
|  | ff17ac53df | ||
|  | 55ba4356f2 | ||
|  | 804b0ff2f2 | ||
|  | 818541d4d7 | ||
|  | 1b199730d2 | ||
|  | f8f685193d | ||
|  | 6e535f8cef | ||
|  | 23340c46e6 | ||
|  | 5c15f5fe04 | ||
|  | ba7e9ed788 | ||
|  | 9ccc8cfb25 | ||
|  | 9810bde68b | ||
|  | 251798a778 | ||
|  | 91f4ccf087 | ||
|  | 73306a1533 | ||
|  | b3eb629785 | ||
|  | aa0b5466a9 | ||
|  | becee5e393 | ||
|  | 59e66dfce5 | ||
|  | 9c59df5e9c | ||
|  | 2a88b25712 | ||
|  | b952f814c1 | ||
|  | f90e3817e8 | ||
|  | 6d6f8b28d7 | ||
|  | 118d56fc40 | ||
|  | f352c34136 | ||
|  | fbf90e6981 | ||
|  | 607faace07 | ||
|  | 521109d3f2 | ||
|  | ec25bdb9f9 | ||
|  | 685962545a | ||
|  | 34d29e7a10 | ||
|  | 05f3e60366 | ||
|  | 5aa1220e5a | ||
|  | c1919c6b24 | ||
|  | 6349e2e28c | ||
|  | e642365613 | ||
|  | ac4086d0ac | ||
|  | d5ff301d90 | ||
|  | 575d4ab431 | ||
|  | ede2be1f66 | ||
|  | d134a993d0 | ||
|  | 86cc6df374 | ||
|  | 32920ca65c | ||
|  | 3ac708ddcb | ||
|  | 0dac0c3a5b | ||
|  | 9810120aff | ||
|  | ae6e844143 | ||
|  | a34e1c0747 | ||
|  | c29ed24a06 | ||
|  | 619621f239 | ||
|  | ff5cd9b592 | ||
|  | af855ef7b4 | ||
|  | 6559d63d3c | ||
|  | 4758cc0c8e | ||
|  | e4ed829661 | ||
|  | 2968e5b61b | ||
|  | 7d274e8088 | ||
|  | 6c2eb6eef3 | ||
|  | 95257d2ee1 | ||
|  | 707d355d4a | ||
|  | 73ba7ed2d2 | ||
|  | 55addfefc8 | ||
|  | 0ecd85cc66 | ||
|  | a9fe3f98c5 | ||
|  | 77b2a88819 | ||
|  | 44621ad28c | ||
|  | 232e2c1e7d | ||
|  | ad3625bef3 | ||
|  | 7c4bf602f0 | ||
|  | ffdd693ff6 | ||
|  | 85b0a47fe8 | ||
|  | 78822a8015 | ||
|  | 55cef21fbe | ||
|  | 2691ac1307 | ||
|  | a51851247e | ||
|  | 0e532a3634 | ||
|  | 883422dc21 | ||
|  | c9daf16388 | ||
|  | b22945e185 | ||
|  | 71150bcaaf | ||
|  | 8c56d1a338 | ||
|  | a49b537d9c | ||
|  | 45328ab719 | ||
|  | 4b755dc58d | ||
|  | 0f29e893f4 | ||
|  | e3adf93a74 | ||
|  | 0d7d5a0318 | ||
|  | 81f88dd998 | ||
|  | b6bfa905db | ||
|  | c0b0f3f0f7 | ||
|  | 16d7b89cb1 | ||
|  | a4560fa20d | ||
|  | fbdb6e6e78 | ||
|  | 8d58f33a28 | ||
|  | 9398222db7 | ||
|  | d2a2362be5 | ||
|  | 4c0a3721d0 | ||
|  | ba2d09f6fb | ||
|  | 7243e65b51 | ||
|  | 3bf4a8fbe2 | ||
|  | 23a6602cbf | ||
|  | 822b94c45d | ||
|  | 0a776c3fd5 | ||
|  | d7378a96ad | ||
|  | db4c6111fd | ||
|  | 2da7fa0397 | ||
|  | 0d58e8d1ad | ||
|  | dad76e0478 | ||
|  | 79aab5aab8 | ||
|  | b02c651961 | ||
|  | 0617a1b0e0 | ||
|  | 06749e71f2 | ||
|  | 6622027c7c | ||
|  | 401c171bbd | ||
|  | a1e766e180 | ||
|  | 63bb770b9c | ||
|  | b3de9a040b | ||
|  | a59dbc4c79 | ||
|  | 40deefa868 | ||
|  | 491de0cf64 | ||
|  | c7b24f4e9c | ||
|  | 27a7563e33 | ||
|  | 25725e9b2f | ||
|  | 819de02101 | ||
|  | ce851a5929 | ||
|  | 7e390ef516 | ||
|  | fb23bd5d26 | ||
|  | 6974f54bfd | ||
|  | aaf5aa4506 | ||
|  | 371b6e3c86 | ||
|  | 9297055ad8 | ||
|  | 9e96089da6 | ||
|  | a79868fadc | ||
|  | 84a0810546 | ||
|  | d9fbb5e25c | ||
|  | e97aa6515b | ||
|  | 6bcfba43c8 | ||
|  | 0c83ee736c | ||
|  | ca55dfe1c6 | ||
|  | 4da33c2bc2 | ||
|  | 2d56be0ebb | ||
|  | 5780dc2b15 | ||
|  | 764bf59d4d | ||
|  | 6742dd8454 | ||
|  | 3ac755bd2f | ||
|  | 7543709ecf | ||
|  | 3ed72c4e46 | ||
|  | 477fa15859 | ||
|  | 1048348ae6 | ||
|  | 390eb9cb61 | ||
|  | 5a1c936ede | ||
|  | 47ad6538f1 | ||
|  | 9be44d8330 | ||
|  | a4b354b33f | ||
|  | a70b864c55 | ||
|  | 3bd5fc0f90 | ||
|  | aabfb792af | ||
|  | e5e48d1cc1 | ||
|  | 42a110dd69 | ||
|  | 64af364b02 | ||
|  | cf14b8fa92 | ||
|  | e7dc6ec025 | ||
|  | f29e311b73 | ||
|  | a914ce2bd2 | ||
|  | b42a7c89e7 | ||
|  | 67483c1b17 | ||
|  | 4071f1e7f2 | ||
|  | 577709fff3 | ||
|  | 8cd45476ac | ||
|  | cf14504fd5 | ||
|  | b84829336d | ||
|  | ba822acb23 | ||
|  | d969e59911 | ||
|  | 936b6148ff | ||
|  | a9776ceafc | ||
|  | e471239955 | ||
|  | 2e8156bfaa | ||
|  | f5dd233a3b | ||
|  | 48ce6c32c1 | ||
|  | 4990239855 | ||
|  | 5e2c929322 | ||
|  | 2b5355c849 | ||
|  | f21f71786a | ||
|  | fc7f109cb2 | ||
|  | a711f0d037 | ||
|  | 98fc6ca441 | ||
|  | c10f1a3a36 | ||
|  | da092e653d | ||
|  | bf29417136 | ||
|  | 79a14ce992 | ||
|  | 99ce26f7b1 | ||
|  | 16250361c3 | ||
|  | be44385b42 | ||
|  | 54c77ecb54 | ||
|  | a30f0dcabd | ||
|  | efef7dce4f | ||
|  | 1c9e4c6050 | ||
|  | 89cd9e8ddd | ||
|  | 92093a8c09 | ||
|  | d970813c20 | ||
|  | f69982aa9d | ||
|  | 82fdc569c2 | ||
|  | def0c1a526 | ||
|  | 93de7cf0c0 | ||
|  | ef2d03d96e | ||
|  | 321c9421ea | ||
|  | 5a225b4196 | ||
|  | 95fabeae73 | ||
|  | 525a6cf5b2 | ||
|  | 27ec0912d5 | ||
|  | 83a7f10c75 | ||
|  | 0a5c9095ac | ||
|  | 0a31225e65 | ||
|  | db4a92d877 | ||
|  | 9df053e3f5 | ||
|  | 1f17731369 | ||
|  | 8e32d1913b | ||
|  | e10a82a501 | ||
|  | ce47f200d5 | ||
|  | 95dc43ce4a | ||
|  | d91eefa74f | ||
|  | ffdfc13461 | ||
|  | a13b03ef3d | ||
|  | 69d504c905 | ||
|  | bda7e025a2 | ||
|  | 596f04eae8 | ||
|  | b39d226fb8 | ||
|  | 20dfb91948 | ||
|  | e033355225 | ||
|  | 56ed45ae70 | ||
|  | d3ff0c2cd4 | ||
|  | 566b205758 | ||
|  | b537ccdb0c | ||
|  | d9b8435a7d | ||
|  | c0ba4d177f | ||
|  | 7377ab7b95 | ||
|  | 207ac94ed0 | ||
|  | fe32a7e584 | ||
|  | 25e12aee14 | ||
|  | 85dd45cb81 | ||
|  | 32340252b2 | ||
|  | 5d716f0149 | ||
|  | 918a343557 | ||
|  | 969dd088a2 | ||
|  | 89001ae9a4 | ||
|  | c99221fa34 | ||
|  | 9ef3fc84f9 | ||
|  | d28bcf24e5 | ||
|  | 8d739c411b | ||
|  | 46c1600ada | ||
|  | 126b32c579 | ||
|  | 380514941c | ||
|  | 61ceb7a32c | ||
|  | 07a3c37a23 | ||
|  | c7e13eb082 | ||
|  | 6906a022ca | ||
|  | 8f0832d340 | ||
|  | bda0dba131 | ||
|  | 76867e39ea | ||
|  | 6f8e8ea252 | ||
|  | 8e7881094f | ||
|  | 7d09132a5c | ||
|  | 6f4a7fb604 | ||
|  | 6e28db513c | ||
|  | 2084201c8f | ||
|  | 70359e5d27 | ||
|  | a72d124551 | ||
|  | 7ff13c3e3e | ||
|  | 55360c1eaf | ||
|  | 60ff50a675 | ||
|  | ba3967aa16 | ||
|  | fffa413121 | ||
|  | c011bdfdd8 | ||
|  | 4235cef1b2 | ||
|  | 871e04cb12 | ||
|  | 287cebb498 | ||
|  | 6c8d200373 | ||
|  | 0ac6f80b50 | ||
|  | 2b73860ea5 | ||
|  | ddcb003b3b | ||
|  | be52c5abb1 | ||
|  | f81ceaef8a | ||
|  | eb6c5fc34d | ||
|  | 4fc16f26a3 | ||
|  | 234d35f592 | ||
|  | 352a72a5d7 | ||
|  | 4d1ce986a6 | ||
|  | 531a8ff248 | ||
|  | 2644c1f598 | ||
|  | fa53f7ec85 | ||
|  | e05574af58 | ||
|  | fcfc976b13 | ||
|  | 78180a5fa7 | ||
|  | 3445abe7ac | ||
|  | e0b442a48b | ||
|  | bd1c84755b | ||
|  | a7194e96e0 | ||
|  | 2bd60f9e60 | ||
|  | 35a40c8727 | ||
|  | 7f62667569 | ||
|  | fd4ba585ee | ||
|  | 81eb46e36d | ||
|  | b7700e77bf | ||
|  | e73dd31619 | ||
|  | 187ec26d8e | ||
|  | ef9b79f85c | ||
|  | 32d88a977d | ||
|  | 547c380961 | ||
|  | 848e23b489 | ||
|  | d63cb1b4d6 | ||
|  | c45de0d8bc | ||
|  | 5c18967f06 | ||
|  | e78f172f02 | ||
|  | 4fc077a5d2 | ||
|  | 7f307d60c4 | ||
|  | b386964abc | ||
|  | 817ac8f256 | ||
|  | c76d58d532 | ||
|  | 4b456f3b76 | ||
|  | 319e3065f0 | ||
|  | a48a8a97a1 | ||
|  | 8be434aaad | ||
|  | d9fc775084 | ||
|  | f25139424a | ||
|  | 2d95c37ea4 | ||
|  | e12630ef06 | ||
|  | 48bd279311 | ||
|  | 36ffdf548d | ||
|  | a5b169c563 | ||
|  | bc5e621683 | ||
|  | 1e69939532 | ||
|  | d8156ef625 | ||
|  | c2c4dc9b58 | ||
|  | ffd4e207a4 | ||
|  | bd3271aff0 | ||
|  | 0664f5a9ca | ||
|  | c515ace328 | ||
|  | 8d4620dc53 | ||
|  | 16f65f669b | ||
|  | 2a2f7f783f | ||
|  | 6ae50389e6 | ||
|  | 87fd51d7ec | ||
|  | 7e43e5615e | ||
|  | 985f8778e9 | ||
|  | 3a180e2afc | ||
|  | 2f47bb0df6 | ||
|  | 7e0f0d9d11 | ||
|  | e1f5866989 | ||
|  | 3c1ed0d9b2 | ||
|  | 10ab39c33b | ||
|  | 3072354ca5 | ||
|  | 14499cd6e5 | ||
|  | 5d3dc3348e | ||
|  | ca2ff214c4 | ||
|  | f8db285d5d | ||
|  | 1f880662d6 | ||
|  | febab86682 | ||
|  | 8070dfef45 | ||
|  | fc69f882c5 | ||
|  | 838a8e18d3 | ||
|  | 5e3e47b484 | ||
|  | 6d8512bda0 | ||
|  | cd68cbd3ea | ||
|  | 55845c95bb | ||
|  | a243ac4dde | ||
|  | a01cbb42c7 | ||
|  | b5da5760a2 | ||
|  | c190b160e9 | ||
|  | ce2e02b690 | ||
|  | 5dab09c42b | ||
|  | 03b08d67f0 | ||
|  | 5841c9a7a5 | ||
|  | ed9b1bea3f | ||
|  | dca348359b | ||
|  | cf0759a48f | ||
|  | c9df233d24 | ||
|  | 99a23b0414 | ||
|  | 95e0633b2f | ||
|  | 5ca210fa60 | ||
|  | 2ccdc419d0 | ||
|  | 9af0e705a5 | ||
|  | 0a3e40332a | ||
|  | a758d18e51 | ||
|  | f15d05b22f | ||
|  | fc9f41b955 | ||
|  | fd1eae4f07 | ||
|  | 51ee77b96f | ||
|  | b03c5ff5ce | ||
|  | 521fed1fea | ||
|  | 679def0151 | ||
|  | 2560626419 | ||
|  | e5024d5d0a | ||
|  | c10c7619d3 | ||
|  | dd04c432e9 | ||
|  | b1fd3b8fc7 | ||
|  | 456df0fc19 | ||
|  | 526f493e12 | ||
|  | 5632ee6378 | ||
|  | 1680f00091 | ||
|  | 376b6f90d9 | ||
|  | 21c0195d29 | ||
|  | 56f845c71a | ||
|  | d6d639d4d7 | ||
|  | e1e1fd640c | ||
|  | 2408eeceba | ||
|  | 6ae194934d | ||
|  | 63ef0f1cee | ||
|  | de2437cfec | ||
|  | 32e08f3510 | ||
|  | 40f21f41e1 | ||
|  | ee12424795 | ||
|  | 0b48d5d0d2 | ||
|  | 080cf98e51 | ||
|  | dc8d5ef744 | ||
|  | 70a02158e5 | ||
|  | ab71dad51a | ||
|  | 0624cefc10 | ||
|  | 56b26421a5 | ||
|  | ea8ba87aeb | ||
|  | 08b258a2cb | ||
|  | ac486d3d1d | ||
|  | e096bf6b62 | ||
|  | e28b33b53b | ||
|  | 5814ba5322 | ||
|  | be81ce244e | ||
|  | d3a3aeb0fc | ||
|  | fe6acdf4d2 | ||
|  | 702e0a461a | ||
|  | 46d6da4fce | ||
|  | aa61835b78 | ||
|  | 2a1e46c8b6 | ||
|  | cb4fb973b2 | ||
|  | 513f6e9a68 | ||
|  | ad980334d1 | ||
|  | d13d078351 | ||
|  | 947798b44c | ||
|  | ed427616d4 | ||
|  | 297921182c | ||
|  | 31a5f3591f | ||
|  | 32655b5b16 | ||
|  | 8947f85ddd | ||
|  | a513a05b7a | ||
|  | 1e716a93ff | ||
|  | 06fc2c505f | ||
|  | 6fcea91d1f | ||
|  | 93d099a2f0 | ||
|  | 29908098e4 | ||
|  | e5983d96f7 | ||
|  | 08e6ae07af | ||
|  | 49b46a9a3f | ||
|  | 36c316f39c | ||
|  | 7e76abc067 | ||
|  | 702e301990 | ||
|  | b1e11f3e88 | ||
|  | 09d5f59701 | ||
|  | 3c8675bb8b | ||
|  | 71ca237478 | ||
|  | 0e4b6d36fd | ||
|  | e898080460 | ||
|  | bdba7d3adf | ||
|  | 606b43dc51 | ||
|  | 2e7833df49 | ||
|  | ec0d03658d | ||
|  | 992d4c1b94 | ||
|  | d2d7cf14e5 | ||
|  | e658712d53 | ||
|  | 40cd6ada4f | ||
|  | c843c182e4 | ||
|  | c35a8bdb15 | ||
|  | dd0701dd16 | ||
|  | 32500773b8 | ||
|  | e7d3f4316f | ||
|  | 438eec720a | ||
|  | 4b38d7368f | ||
|  | dce6a86900 | ||
|  | dc9c558c06 | ||
|  | b8a466c571 | ||
|  | bae28c5f57 | ||
|  | 1b21f0723f | ||
|  | 911c439858 | ||
|  | f81f85cea2 | ||
|  | 1325cc5cd0 | ||
|  | 951d61bfcd | ||
|  | 0937cba870 | ||
|  | 5597d7633d | ||
|  | 502c88ee3f | ||
|  | 5ef6297daa | ||
|  | 9e33e23b8b | ||
|  | 16d00ccffb | ||
|  | d211437d6c | ||
|  | 7996a42f76 | ||
|  | f482e5e84a | ||
|  | 447c3567b4 | ||
|  | 3c5e6fe7f8 | ||
|  | bf4a578bbb | ||
|  | 4cabea069d | ||
|  | c53033a778 | ||
|  | ea8642e2a1 | ||
|  | 73cea2d303 | ||
|  | 96a3468791 | ||
|  | 2065f4c003 | ||
|  | 9a931e4dc9 | ||
|  | 49ec62c757 | ||
|  | a371f971fb | ||
|  | 5f9a84fc8b | ||
|  | 2461e36ed4 | ||
|  | 1305bf49a5 | ||
|  | da0a16e122 | ||
|  | fb10687168 | ||
|  | f0d78471af | ||
|  | a90b2a672e | ||
|  | 2bbb6fc427 | ||
|  | 2747e240c1 | ||
|  | 4b370930b5 | ||
|  | c74918321d | ||
|  | b05a5c818d | ||
|  | 41d22ef17e | ||
|  | bbee63fcf3 | ||
|  | b1ddd0e038 | ||
|  | 8c5dc3b5cb | ||
|  | afa05329d9 | ||
|  | dbbff393e1 | ||
|  | f742671bbe | ||
|  | 0dae829080 | ||
|  | e62a00a3f5 | ||
|  | ab4c93dd2f | ||
|  | ed5321999c | ||
|  | fb21e3bb5c | ||
|  | 3595292f7f | ||
|  | 47fb6e036a | ||
|  | 92886c46ea | ||
|  | 83fa3f4cc8 | ||
|  | c24f75ce0b | ||
|  | 63929b0341 | ||
|  | fc7ec17905 | ||
|  | e5a01c7cc8 | ||
|  | 0509b6fdb9 | ||
|  | 60d87f3c64 | ||
|  | 5d800ba5fe | ||
|  | 759d17547a | ||
|  | d4f0a9ff62 | ||
|  | c4fa96c41e | ||
|  | f54136b602 | ||
|  | 5dd1728bf8 | ||
|  | da1c9f48b7 | ||
|  | 0ec0e37532 | ||
|  | 544dc2eaa5 | ||
|  | a3327c4430 | ||
|  | f8ae972e70 | ||
|  | 3ff83fc1f8 | ||
|  | 63f65e5b2a | ||
|  | 3140a4e0cd | ||
|  | 31038e0e12 | ||
|  | ac8e47579b | ||
|  | ec0075e0d0 | ||
|  | 7900d266b1 | ||
|  | c21597c593 | ||
|  | ea418aa7d8 | ||
|  | 5487015a83 | ||
|  | 418cccd307 | ||
|  | 2a0760412c | ||
|  | eebbe64b36 | ||
|  | 42d8e6d60d | ||
|  | 7ba907f261 | ||
|  | c72769e2ea | ||
|  | 02d856b8a5 | ||
|  | 0d15ac8861 | ||
|  | 134a767a7f | ||
|  | 7403b6fb82 | ||
|  | 64a65cadf3 | ||
|  | 121eaced49 | ||
|  | a488430f23 | ||
|  | b5db753e11 | ||
|  | b0aa27db31 | ||
|  | 512ed086bd | ||
|  | 76e35a09b7 | ||
|  | d2c1d39d42 | ||
|  | e9cccf6504 | ||
|  | 1c505903ff | ||
|  | 53ed8e04ae | ||
|  | 2112de6f15 | ||
|  | be0845af02 | ||
|  | f83a57b3da | ||
|  | 08264749f0 | ||
|  | a75819cae3 | ||
|  | 9fb32a47ca | ||
|  | 4f43c9ebb4 | ||
|  | 9177982334 | ||
|  | 84b125bdde | ||
|  | 52eeff9f9f | ||
|  | 0fcccd35ff | ||
|  | 598dcf6b62 | ||
|  | 459200dd01 | ||
|  | af22cabc6f | ||
|  | 920e82f11a | ||
|  | 520fcf82ae | ||
|  | 9bdf9e1e02 | ||
|  | 3a45f05e36 | ||
|  | 8e3e387be7 | ||
|  | 267d0b7b5a | ||
|  | 74d1d55051 | ||
|  | 3a8cb3f010 | ||
|  | f5b290b093 | ||
|  | d38d11f02e | ||
|  | af04e92cf2 | ||
|  | 4ea1c98ac9 | ||
|  | 05333b9579 | ||
|  | 49cdb67ddc | ||
|  | b5198e63c4 | ||
|  | db007efe00 | ||
|  | 699cf71652 | ||
|  | a0c02f62a3 | ||
|  | ff7b814edc | ||
|  | 015f24a901 | ||
|  | 4fccde84bd | ||
|  | ea459e9af0 | ||
|  | 2dd5a53db2 | ||
|  | fc97ea7ee0 | ||
|  | 582d2540af | ||
|  | 6ad79dcd45 | ||
|  | 721896ba70 | ||
|  | 228270414c | ||
|  | 2683df7b5b | ||
|  | 3e61d1f233 | ||
|  | 04c07227f2 | ||
|  | 2e8d99c5b8 | ||
|  | c07301473b | ||
|  | b1ba42410b | ||
|  | b80f89e3db | ||
|  | edb15a9346 | ||
|  | 714a4d4f2d | ||
|  | 5c853766e8 | ||
|  | 3567ae88ad | ||
|  | afcec56be4 | ||
|  | d2435cf43b | ||
|  | 556f7608db | ||
|  | a4df4b028e | ||
|  | 63683d35fc | ||
|  | 495344591f | ||
|  | 4e508499da | ||
|  | 326be29568 | ||
|  | e4a3df3516 | ||
|  | 3506cbd5e9 | ||
|  | ab13019bde | ||
|  | ddc663eac0 | ||
|  | fc7002fbab | ||
|  | f2e53a3569 | ||
|  | c5b4e589ff | ||
|  | 5e63ab619e | ||
|  | c9bbfa1272 | ||
|  | 050968cbac | ||
|  | 8ca0d804d8 | ||
|  | 54e5a3607e | ||
|  | cd947ae822 | ||
|  | 2477e18c87 | ||
|  | ef08e8b8a0 | ||
|  | f59bf16e82 | ||
|  | 118c31eb8d | ||
|  | 476f16f0aa | ||
|  | b40d35b779 | ||
|  | 8e016cf672 | ||
|  | 7e482e9f8b | ||
|  | 6445befe87 | ||
|  | 86c099d629 | ||
|  | 79af433381 | ||
|  | c0f1e74bed | ||
|  | 9df89e66e3 | ||
|  | 660375d6e4 | ||
|  | 498e8545b6 | ||
|  | 230c2e5cc2 | ||
|  | 3e60863e2d | ||
|  | 4592626bbb | ||
|  | b980c87eff | ||
|  | 0f7c322623 | ||
|  | 76f42a3013 | ||
|  | 93b3d601d5 | ||
|  | 56329e89bb | ||
|  | 5c8b8149eb | ||
|  | 6075f7e8fd | ||
|  | ddf53494f0 | ||
|  | cd1f03d4f4 | ||
|  | 8474a61f21 | ||
|  | 4ad0ab5433 | ||
|  | 66d151df77 | ||
|  | 2045b250fd | ||
|  | 1dbee90d34 | ||
|  | eb7a6d925b | ||
|  | 3678bd5a93 | ||
|  | 2d1a973ee5 | ||
|  | 322f7b2ad4 | ||
|  | 41aa2672cd | ||
|  | f3090a452a | ||
|  | 52790d3c37 | ||
|  | 3677252e17 | ||
|  | 235d1d655d | ||
|  | 29bd6faa18 | ||
|  | 69c0f38305 | ||
|  | 0399d0c4d6 | ||
|  | 3db47f0adc | ||
|  | 483e2c43cf | ||
|  | 3e3b7238e0 | ||
|  | 532b5865de | ||
|  | 54b94f29e1 | ||
|  | b67a7215f6 | ||
|  | e424cc7608 | ||
|  | 229008e76a | ||
|  | 584f4bc596 | ||
|  | 1502d20def | ||
|  | eecc2f4dd7 | ||
|  | 6fc110a71a | ||
|  | ca6b46533a | ||
|  | a1fe29347a | ||
|  | 449afea4fc | ||
|  | 6e5dd35ee3 | ||
|  | 0d5d14d41a | ||
|  | 3a42e457cf | ||
|  | 5b05c990b0 | ||
|  | 9df0a6208b | ||
|  | 3214904cc7 | ||
|  | ec775a016a | ||
|  | a2ca235fee | ||
|  | de458b7357 | ||
|  | 7c039ca223 | ||
|  | 3942962ef5 | ||
|  | 675655d437 | ||
|  | dafb14ff37 | ||
|  | fc52d1cfba | ||
|  | fdf2a68a11 | ||
|  | 3908ef611a | ||
|  | e63db782c1 | ||
|  | a6c6127e33 | ||
|  | 207d0bec78 | ||
|  | 1443c8d4c6 | ||
|  | a136c46148 | ||
|  | bbbc18fd84 | ||
|  | 2c7f6e4def | ||
|  | dcd0cda0c6 | ||
|  | ff16925f63 | ||
|  | 0b7aaa3643 | ||
|  | 44a244b1cb | ||
|  | 1dc6f39b55 | ||
|  | 45f52ca29c | ||
|  | fae2d93525 | ||
|  | 25b74ce1f3 | ||
|  | 4957e498af | ||
|  | 54ca1abd2b | ||
|  | 8f2951b275 | ||
|  | 720bef97e6 | ||
|  | c42f1b7a50 | ||
|  | 0186c31d59 | ||
|  | 58bf1a2ca5 | ||
|  | 4a31544024 | ||
|  | cb6ec507e2 | ||
|  | 1ef93fead7 | ||
|  | 285ded6e49 | ||
|  | 6e4f5821dc | ||
|  | a3df5b9a94 | ||
|  | 04f0ebf776 | ||
|  | 0e97a3becd | ||
|  | 77a0cef9ce | ||
|  | 143e9b6f9c | ||
|  | 06dcf8d8aa | ||
|  | c315b4e064 | ||
|  | 73ca7ad0c1 | ||
|  | d7f517fbf5 | ||
|  | b10cb84f33 | ||
|  | ed216bea4d | ||
|  | 7669f41e8e | ||
|  | 73513f8371 | ||
|  | cb1d0441e9 | ||
|  | 7affeae480 | ||
|  | 7928e6d0cd | ||
|  | a98b726263 | ||
|  | 5f0b6fde92 | ||
|  | 6b1158235e | ||
|  | f624449ccb | 
							
								
								
									
										24
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										24
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
								
							| @@ -1,24 +0,0 @@ | ||||
| provider/kubernetes/**  @containous/kubernetes | ||||
| provider/rancher/**     @containous/rancher | ||||
| provider/marathon/**    @containous/marathon | ||||
| provider/docker/**      @containous/docker | ||||
|  | ||||
| docs/user-guide/kubernetes.md       @containous/kubernetes | ||||
| docs/user-guide/marathon.md         @containous/marathon | ||||
| docs/user-guide/swarm.md            @containous/docker | ||||
| docs/user-guide/swarm-mode.md       @containous/docker | ||||
|  | ||||
| docs/configuration/backends/docker.md       @containous/docker | ||||
| docs/configuration/backends/kubernetes.md   @containous/kubernetes | ||||
| docs/configuration/backends/marathon.md     @containous/marathon | ||||
| docs/configuration/backends/rancher.md      @containous/rancher | ||||
|  | ||||
| examples/k8s/                   @containous/kubernetes | ||||
| examples/compose-k8s.yaml       @containous/kubernetes | ||||
| examples/k8s.namespace.yaml     @containous/kubernetes | ||||
| examples/compose-rancher.yml    @containous/rancher | ||||
| examples/compose-marathon.yml   @containous/marathon | ||||
|  | ||||
| vendor/github.com/gambol99/go-marathon  @containous/marathon | ||||
| vendor/github.com/rancher               @containous/rancher | ||||
| vendor/k8s.io/                          @containous/kubernetes | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -8,7 +8,7 @@ DO NOT FILE ISSUES FOR GENERAL SUPPORT QUESTIONS. | ||||
| The issue tracker is for reporting bugs and feature requests only. | ||||
| For end-user related support questions, please refer to one of the following: | ||||
|  | ||||
| - the Traefik community forum: https://community.containo.us/ | ||||
| - the Traefik community forum: https://community.traefik.io/ | ||||
|  | ||||
| --> | ||||
|  | ||||
| @@ -17,7 +17,7 @@ Bug | ||||
| <!-- | ||||
|  | ||||
| The configurations between 1.X and 2.X are NOT compatible. | ||||
| Please have a look here https://docs.traefik.io/v2.0/getting-started/configuration-overview/. | ||||
| Please have a look here https://doc.traefik.io/traefik/getting-started/configuration-overview/. | ||||
|  | ||||
| --> | ||||
|  | ||||
|   | ||||
							
								
								
									
										82
									
								
								.github/ISSUE_TEMPLATE/Bug_report.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										82
									
								
								.github/ISSUE_TEMPLATE/Bug_report.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,82 +0,0 @@ | ||||
| --- | ||||
| name: Bug report | ||||
| about: Create a report to help us improve | ||||
|  | ||||
| --- | ||||
| <!-- PLEASE FOLLOW THE ISSUE TEMPLATE TO HELP TRIAGE AND SUPPORT! --> | ||||
|  | ||||
| ### Do you want to request a *feature* or report a *bug*? | ||||
|  | ||||
| <!-- | ||||
| DO NOT FILE ISSUES FOR GENERAL SUPPORT QUESTIONS. | ||||
|  | ||||
| The issue tracker is for reporting bugs and feature requests only. | ||||
| For end-user related support questions, please refer to one of the following: | ||||
|  | ||||
| - the Traefik community forum: https://community.containo.us/ | ||||
|  | ||||
| --> | ||||
|  | ||||
| Bug | ||||
|  | ||||
| <!-- | ||||
|  | ||||
| The configurations between 1.X and 2.X are NOT compatible. | ||||
| Please have a look here https://docs.traefik.io/v2.0/getting-started/configuration-overview/. | ||||
|  | ||||
| --> | ||||
|  | ||||
| ### What did you do? | ||||
|  | ||||
| <!-- | ||||
|  | ||||
| HOW TO WRITE A GOOD BUG REPORT? | ||||
|  | ||||
| - Respect the issue template as much as possible. | ||||
| - The title should be short and descriptive. | ||||
| - Explain the conditions which led you to report this issue: the context. | ||||
| - The context should lead to something, an idea or a problem that you’re facing. | ||||
| - Remain clear and concise. | ||||
| - Format your messages to help the reader focus on what matters and understand the structure of your message, use Markdown syntax https://help.github.com/articles/github-flavored-markdown | ||||
|  | ||||
| --> | ||||
|  | ||||
| ### What did you expect to see? | ||||
|  | ||||
|  | ||||
|  | ||||
| ### What did you see instead? | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Output of `traefik version`: (_What version of Traefik are you using?_) | ||||
|  | ||||
| <!-- | ||||
| `latest` is not considered as a valid version. | ||||
|  | ||||
| For the Traefik Docker image: | ||||
|     docker run [IMAGE] version | ||||
|     ex: docker run traefik version | ||||
|  | ||||
| --> | ||||
|  | ||||
| ``` | ||||
| (paste your output here) | ||||
| ``` | ||||
|  | ||||
| ### What is your environment & configuration (arguments, toml, provider, platform, ...)? | ||||
|  | ||||
| ```toml | ||||
| # (paste your configuration here) | ||||
| ``` | ||||
|  | ||||
| <!-- | ||||
| Add more configuration information here. | ||||
| --> | ||||
|  | ||||
|  | ||||
| ### If applicable, please paste the log output in DEBUG level (`--log.level=DEBUG` switch) | ||||
|  | ||||
| ``` | ||||
| (paste your output here) | ||||
| ``` | ||||
							
								
								
									
										35
									
								
								.github/ISSUE_TEMPLATE/Feature_request.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										35
									
								
								.github/ISSUE_TEMPLATE/Feature_request.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,35 +0,0 @@ | ||||
| --- | ||||
| name: Feature request | ||||
| about: Suggest an idea for this project | ||||
|  | ||||
| --- | ||||
| <!-- PLEASE FOLLOW THE ISSUE TEMPLATE TO HELP TRIAGE AND SUPPORT! --> | ||||
|  | ||||
| ### Do you want to request a *feature* or report a *bug*? | ||||
|  | ||||
| <!-- | ||||
| DO NOT FILE ISSUES FOR GENERAL SUPPORT QUESTIONS. | ||||
|  | ||||
| The issue tracker is for reporting bugs and feature requests only. | ||||
| For end-user related support questions, please refer to one of the following: | ||||
|  | ||||
| - the Traefik community forum: https://community.containo.us/ | ||||
|  | ||||
| --> | ||||
|  | ||||
| Feature | ||||
|  | ||||
| ### What did you expect to see? | ||||
|  | ||||
| <!-- | ||||
|  | ||||
| HOW TO WRITE A GOOD ISSUE? | ||||
|  | ||||
| - Respect the issue template as much as possible. | ||||
| - The title should be short and descriptive. | ||||
| - Explain the conditions which led you to report this issue: the context. | ||||
| - The context should lead to something, an idea or a problem that you’re facing. | ||||
| - Remain clear and concise. | ||||
| - Format your messages to help the reader focus on what matters and understand the structure of your message, use Markdown syntax https://help.github.com/articles/github-flavored-markdown | ||||
|  | ||||
| --> | ||||
							
								
								
									
										82
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| name: Bug Report (Traefik) | ||||
| description: Create a report to help us improve. | ||||
| body: | ||||
|   - type: checkboxes | ||||
|     id: terms | ||||
|     attributes: | ||||
|       label: Welcome! | ||||
|       description: | | ||||
|         The issue tracker is for reporting bugs and feature requests only. | ||||
|         For end-user related support questions, please use the [Traefik community forum](https://community.traefik.io/). | ||||
|  | ||||
|         All new/updated issues are triaged regularly by the maintainers. | ||||
|         All issues closed by a bot are subsequently double-checked by the maintainers. | ||||
|  | ||||
|         DO NOT FILE ISSUES FOR GENERAL SUPPORT QUESTIONS. | ||||
|  | ||||
|       options: | ||||
|         - label: Yes, I've searched similar issues on [GitHub](https://github.com/traefik/traefik/issues) and didn't find any. | ||||
|           required: true | ||||
|         - label: Yes, I've searched similar issues on the [Traefik community forum](https://community.traefik.io) and didn't find any. | ||||
|           required: true | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: What did you do? | ||||
|       description: | | ||||
|         How to write a good bug report? | ||||
|  | ||||
|         - Respect the issue template as much as possible. | ||||
|         - The title should be short and descriptive. | ||||
|         - Explain the conditions which led you to report this issue: the context. | ||||
|         - The context should lead to something, an idea or a problem that you’re facing. | ||||
|         - Remain clear and concise. | ||||
|         - Format your messages to help the reader focus on what matters and understand the structure of your message, use [Markdown syntax](https://help.github.com/articles/github-flavored-markdown) | ||||
|       placeholder: What did you do? | ||||
|     validations: | ||||
|       required: true | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: What did you see instead? | ||||
|       placeholder: What did you see instead? | ||||
|     validations: | ||||
|       required: true | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: What version of Traefik are you using? | ||||
|       description: | | ||||
|         `latest` is not considered as a valid version. | ||||
|  | ||||
|         Output of `traefik version`. | ||||
|  | ||||
|         For the Traefik Docker image (`docker run [IMAGE] version`), example: | ||||
|         ```console | ||||
|         $ docker run traefik version | ||||
|         ``` | ||||
|       placeholder: Paste your output here. | ||||
|     validations: | ||||
|       required: true | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: What is your environment & configuration? | ||||
|       description: arguments, toml, provider, platform, ... | ||||
|       placeholder: Add information here. | ||||
|       value: | | ||||
|         ```yaml | ||||
|         # (paste your configuration here) | ||||
|         ``` | ||||
|  | ||||
|         Add more configuration information here. | ||||
|     validations: | ||||
|       required: true | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: If applicable, please paste the log output in DEBUG level | ||||
|       description: "`--log.level=DEBUG` switch." | ||||
|       placeholder: Paste your output here. | ||||
|     validations: | ||||
|       required: false | ||||
							
								
								
									
										8
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| blank_issues_enabled: false | ||||
| contact_links: | ||||
|   - name: Traefik Community Support | ||||
|     url: https://community.traefik.io/ | ||||
|     about: If you have a question, or are looking for advice, please post on our Discuss forum! The community loves to chime in to help. Happy Coding! | ||||
|   - name: Traefik Helm Chart Issues | ||||
|     url: https://github.com/traefik/traefik-helm-chart | ||||
|     about: Are you submitting an issue or feature enhancement for the Traefik helm chart? Please post in the traefik-helm-chart GitHub Issues. | ||||
							
								
								
									
										33
									
								
								.github/ISSUE_TEMPLATE/feature-request.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								.github/ISSUE_TEMPLATE/feature-request.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| name: Feature Request (Traefik) | ||||
| description: Suggest an idea for this project. | ||||
| body: | ||||
|   - type: checkboxes | ||||
|     id: terms | ||||
|     attributes: | ||||
|       label: Welcome! | ||||
|       description: | | ||||
|         The issue tracker is for reporting bugs and feature requests only. For end-user related support questions, please refer to one of the following: | ||||
|         - the Traefik community forum: https://community.traefik.io/ | ||||
|  | ||||
|         DO NOT FILE ISSUES FOR GENERAL SUPPORT QUESTIONS. | ||||
|       options: | ||||
|         - label: Yes, I've searched similar issues on [GitHub](https://github.com/traefik/traefik/issues) and didn't find any. | ||||
|           required: true | ||||
|         - label: Yes, I've searched similar issues on the [Traefik community forum](https://community.traefik.io) and didn't find any. | ||||
|           required: true | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: What did you expect to see? | ||||
|       description: | | ||||
|         How to write a good issue? | ||||
|  | ||||
|         - Respect the issue template as much as possible. | ||||
|         - The title should be short and descriptive. | ||||
|         - Explain the conditions which led you to report this issue: the context. | ||||
|         - The context should lead to something, an idea or a problem that you’re facing. | ||||
|         - Remain clear and concise. | ||||
|         - Format your messages to help the reader focus on what matters and understand the structure of your message, use [Markdown syntax](https://help.github.com/articles/github-flavored-markdown) | ||||
|       placeholder: What did you expect to see? | ||||
|     validations: | ||||
|       required: true | ||||
							
								
								
									
										14
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -2,18 +2,18 @@ | ||||
| PLEASE READ THIS MESSAGE. | ||||
|  | ||||
| Documentation fixes or enhancements: | ||||
| - for Traefik v1: use branch v1.7 | ||||
| - for Traefik v2: use branch v2.2 | ||||
| - for Traefik v2: use branch v2.11 | ||||
| - for Traefik v3: use branch v3.0 | ||||
|  | ||||
| Bug fixes: | ||||
| - for Traefik v1: use branch v1.7 | ||||
| - for Traefik v2: use branch v2.2 | ||||
| - for Traefik v2: use branch v2.11 | ||||
| - for Traefik v3: use branch v3.0 | ||||
|  | ||||
| Enhancements: | ||||
| - for Traefik v1: we only accept bug fixes | ||||
| - for Traefik v2: use branch master | ||||
| - for Traefik v2: we only accept bug fixes | ||||
| - for Traefik v3: use branch master | ||||
|  | ||||
| HOW TO WRITE A GOOD PULL REQUEST? https://docs.traefik.io/contributing/submitting-pull-requests/ | ||||
| HOW TO WRITE A GOOD PULL REQUEST? https://doc.traefik.io/traefik/contributing/submitting-pull-requests/ | ||||
|  | ||||
| --> | ||||
|  | ||||
|   | ||||
							
								
								
									
										79
									
								
								.github/workflows/build.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								.github/workflows/build.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | ||||
| name: Build Binaries | ||||
|  | ||||
| on: | ||||
|   pull_request: | ||||
|     branches: | ||||
|       - '*' | ||||
|  | ||||
| env: | ||||
|   GO_VERSION: '1.21' | ||||
|   CGO_ENABLED: 0 | ||||
|   IN_DOCKER: "" | ||||
|  | ||||
| jobs: | ||||
|  | ||||
|   build-webui: | ||||
|     runs-on: ubuntu-20.04 | ||||
|  | ||||
|     steps: | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Build webui | ||||
|         run: | | ||||
|           make clean-webui generate-webui | ||||
|           tar czvf webui.tar.gz ./webui/static/ | ||||
|  | ||||
|       - name: Artifact webui | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: webui.tar.gz | ||||
|           path: webui.tar.gz | ||||
|  | ||||
|   build: | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ ubuntu-20.04, macos-latest, windows-latest ] | ||||
|     needs: | ||||
|       - build-webui | ||||
|     defaults: | ||||
|       run: | ||||
|         working-directory: ${{ github.workspace }}/go/src/github.com/traefik/traefik | ||||
|  | ||||
|     steps: | ||||
|       - name: Set up Go ${{ env.GO_VERSION }} | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: ${{ env.GO_VERSION }} | ||||
|  | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           path: go/src/github.com/traefik/traefik | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Cache Go modules | ||||
|         uses: actions/cache@v3 | ||||
|         with: | ||||
|           path: | | ||||
|             ~/go/pkg/mod | ||||
|             ~/.cache/go-build | ||||
|             ~/Library/Caches/go-build | ||||
|             '%LocalAppData%\go-build' | ||||
|           key: ${{ runner.os }}-build-go-${{ hashFiles('**/go.sum') }} | ||||
|           restore-keys: ${{ runner.os }}-build-go- | ||||
|  | ||||
|       - name: Artifact webui | ||||
|         uses: actions/download-artifact@v2 | ||||
|         with: | ||||
|           name: webui.tar.gz | ||||
|           path: ${{ github.workspace }}/go/src/github.com/traefik/traefik | ||||
|  | ||||
|       - name: Untar webui | ||||
|         run: tar xvf webui.tar.gz | ||||
|  | ||||
|       - name: Build | ||||
|         run: make binary | ||||
							
								
								
									
										25
									
								
								.github/workflows/check_doc.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								.github/workflows/check_doc.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| name: Check Documentation | ||||
|  | ||||
| on: | ||||
|   pull_request: | ||||
|     branches: | ||||
|       - '*' | ||||
|  | ||||
| jobs: | ||||
|  | ||||
|   docs: | ||||
|     name: Check, verify and build documentation | ||||
|     runs-on: ubuntu-20.04 | ||||
|  | ||||
|     steps: | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Check documentation | ||||
|         run: make docs-pull-images docs | ||||
|         env: | ||||
|           # These variables are not passed to workflows that are triggered by a pull request from a fork. | ||||
|           DOCS_VERIFY_SKIP: ${{ vars.DOCS_VERIFY_SKIP }} | ||||
|           DOCS_LINT_SKIP: ${{ vars.DOCS_LINT_SKIP }} | ||||
							
								
								
									
										64
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| name: "CodeQL" | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|       - v* | ||||
|   schedule: | ||||
|     - cron: '11 22 * * 1' | ||||
|  | ||||
| jobs: | ||||
|   analyze: | ||||
|     name: Analyze | ||||
|     runs-on: ubuntu-latest | ||||
|     permissions: | ||||
|       actions: read | ||||
|       contents: read | ||||
|       security-events: write | ||||
|  | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         language: [ 'javascript', 'go' ] | ||||
|         # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ] | ||||
|         # Use only 'java' to analyze code written in Java, Kotlin or both | ||||
|         # Use only 'javascript' to analyze code written in JavaScript, TypeScript or both | ||||
|         # Learn more about CodeQL language support at https://aka.ms/codeql-docs/language-support | ||||
|  | ||||
|     steps: | ||||
|     - name: Checkout repository | ||||
|       uses: actions/checkout@v3 | ||||
|  | ||||
|     # Initializes the CodeQL tools for scanning. | ||||
|     - name: Initialize CodeQL | ||||
|       uses: github/codeql-action/init@v2 | ||||
|       with: | ||||
|         languages: ${{ matrix.language }} | ||||
|         # If you wish to specify custom queries, you can do so here or in a config file. | ||||
|         # By default, queries listed here will override any specified in a config file. | ||||
|         # Prefix the list here with "+" to use these queries and those in the config file. | ||||
|  | ||||
|         # For more details on CodeQL's query packs, refer to: https://docs.github.com/en/code-security/code-scanning/automatically-scanning-your-code-for-vulnerabilities-and-errors/configuring-code-scanning#using-queries-in-ql-packs | ||||
|         # queries: security-extended,security-and-quality | ||||
|  | ||||
|  | ||||
|     # Autobuild attempts to build any compiled languages (C/C++, C#, Go, Java, or Swift). | ||||
|     # If this step fails, then you should remove it and run the build manually (see below) | ||||
|     - name: Autobuild | ||||
|       uses: github/codeql-action/autobuild@v2 | ||||
|  | ||||
|     # ℹ️ Command-line programs to run using the OS shell. | ||||
|     # 📚 See https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_idstepsrun | ||||
|  | ||||
|     #   If the Autobuild fails above, remove it and uncomment the following three lines. | ||||
|     #   modify them (or add more) to build your code if your project, please refer to the EXAMPLE below for guidance. | ||||
|  | ||||
|     # - run: | | ||||
|     #     echo "Run, Build Application using script" | ||||
|     #     ./location_of_script_within_repo/buildscript.sh | ||||
|  | ||||
|     - name: Perform CodeQL Analysis | ||||
|       uses: github/codeql-action/analyze@v2 | ||||
|       with: | ||||
|         category: "/language:${{matrix.language}}" | ||||
							
								
								
									
										52
									
								
								.github/workflows/documentation.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								.github/workflows/documentation.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| name: Build and Publish Documentation | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|       - v* | ||||
|  | ||||
| env: | ||||
|   STRUCTOR_VERSION: v1.13.2 | ||||
|   MIXTUS_VERSION: v0.4.1 | ||||
|  | ||||
| jobs: | ||||
|  | ||||
|   docs: | ||||
|     name: Doc Process | ||||
|     runs-on: ubuntu-20.04 | ||||
|     if: github.repository == 'traefik/traefik' | ||||
|  | ||||
|     steps: | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Login to DockerHub | ||||
|         uses: docker/login-action@v1 | ||||
|         with: | ||||
|           username: ${{ secrets.DOCKERHUB_USERNAME }} | ||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||
|  | ||||
|       - name: Install Structor ${{ env.STRUCTOR_VERSION }} | ||||
|         run: curl -sSfL https://raw.githubusercontent.com/traefik/structor/master/godownloader.sh | sh -s -- -b $HOME/bin ${STRUCTOR_VERSION} | ||||
|  | ||||
|       - name: Install Seo-doc | ||||
|         run: curl -sSfL https://raw.githubusercontent.com/traefik/seo-doc/master/godownloader.sh | sh -s -- -b "${HOME}/bin" | ||||
|  | ||||
|       - name: Install Mixtus ${{ env.MIXTUS_VERSION }} | ||||
|         run: curl -sSfL https://raw.githubusercontent.com/traefik/mixtus/master/godownloader.sh | sh -s -- -b $HOME/bin ${MIXTUS_VERSION} | ||||
|  | ||||
|       - name: Build documentation | ||||
|         run: $HOME/bin/structor -o traefik -r traefik --dockerfile-url="https://raw.githubusercontent.com/traefik/traefik/v1.7/docs.Dockerfile" --menu.js-url="https://raw.githubusercontent.com/traefik/structor/master/traefik-menu.js.gotmpl" --rqts-url="https://raw.githubusercontent.com/traefik/structor/master/requirements-override.txt" --force-edit-url --exp-branch=master --debug | ||||
|         env: | ||||
|           STRUCTOR_LATEST_TAG: ${{ vars.STRUCTOR_LATEST_TAG }} | ||||
|  | ||||
|       - name: Apply seo | ||||
|         run: $HOME/bin/seo -path=./site -product=traefik | ||||
|  | ||||
|       - name: Publish documentation | ||||
|         run: $HOME/bin/mixtus --dst-doc-path="./traefik" --dst-owner=traefik --dst-repo-name=doc --git-user-email="30906710+traefiker@users.noreply.github.com" --git-user-name=traefiker --src-doc-path="./site" --src-owner=containous --src-repo-name=traefik | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GH_TOKEN_REPO }} | ||||
							
								
								
									
										37
									
								
								.github/workflows/experimental.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								.github/workflows/experimental.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| name: Build experimental image on branch | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|       - v* | ||||
|  | ||||
| jobs: | ||||
|  | ||||
|   experimental: | ||||
|     if: github.repository == 'traefik/traefik' | ||||
|     name: Build experimental image on branch | ||||
|     runs-on: ubuntu-20.04 | ||||
|  | ||||
|     steps: | ||||
|  | ||||
|       # https://github.com/marketplace/actions/checkout | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Branch name | ||||
|         run: echo ${GITHUB_REF##*/} | ||||
|  | ||||
|       - name: Build docker experimental image | ||||
|         run: docker build -t traefik/traefik:experimental-${GITHUB_REF##*/} -f exp.Dockerfile . | ||||
|  | ||||
|       - name: Login to Docker Hub | ||||
|         uses: docker/login-action@v1 | ||||
|         with: | ||||
|           username: ${{ secrets.DOCKERHUB_USERNAME }} | ||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||
|  | ||||
|       - name: Push to Docker Hub | ||||
|         run: docker push traefik/traefik:experimental-${GITHUB_REF##*/} | ||||
							
								
								
									
										46
									
								
								.github/workflows/test-unit.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								.github/workflows/test-unit.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| name: Test Unit | ||||
|  | ||||
| on: | ||||
|   pull_request: | ||||
|     branches: | ||||
|       - '*' | ||||
|  | ||||
| env: | ||||
|   GO_VERSION: '1.21' | ||||
|   IN_DOCKER: "" | ||||
|  | ||||
| jobs: | ||||
|  | ||||
|   test-unit: | ||||
|     runs-on: ubuntu-20.04 | ||||
|  | ||||
|     defaults: | ||||
|       run: | ||||
|         working-directory: ${{ github.workspace }}/go/src/github.com/traefik/traefik | ||||
|  | ||||
|     steps: | ||||
|       - name: Set up Go ${{ env.GO_VERSION }} | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: ${{ env.GO_VERSION }} | ||||
|  | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           path: go/src/github.com/traefik/traefik | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Cache Go modules | ||||
|         uses: actions/cache@v3 | ||||
|         with: | ||||
|           path: | | ||||
|             ~/go/pkg/mod | ||||
|             ~/.cache/go-build | ||||
|           key: ${{ runner.os }}-test-unit-go-${{ hashFiles('**/go.sum') }} | ||||
|           restore-keys: ${{ runner.os }}-test-unit-go- | ||||
|  | ||||
|       - name: Avoid generating webui | ||||
|         run: touch webui/static/index.html | ||||
|  | ||||
|       - name: Tests | ||||
|         run: make test-unit | ||||
							
								
								
									
										97
									
								
								.github/workflows/validate.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								.github/workflows/validate.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| name: Validate | ||||
|  | ||||
| on: | ||||
|   pull_request: | ||||
|     branches: | ||||
|       - '*' | ||||
|  | ||||
| env: | ||||
|   GO_VERSION: '1.21' | ||||
|   GOLANGCI_LINT_VERSION: v1.55.2 | ||||
|   MISSSPELL_VERSION: v0.4.0 | ||||
|   IN_DOCKER: "" | ||||
|  | ||||
| jobs: | ||||
|  | ||||
|   validate: | ||||
|     runs-on: ubuntu-20.04 | ||||
|  | ||||
|     defaults: | ||||
|       run: | ||||
|         working-directory: ${{ github.workspace }}/go/src/github.com/traefik/traefik | ||||
|  | ||||
|     steps: | ||||
|       - name: Set up Go ${{ env.GO_VERSION }} | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: ${{ env.GO_VERSION }} | ||||
|  | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           path: go/src/github.com/traefik/traefik | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Cache Go modules | ||||
|         uses: actions/cache@v3 | ||||
|         with: | ||||
|           path: | | ||||
|             ~/go/pkg/mod | ||||
|             ~/.cache/go-build | ||||
|           key: ${{ runner.os }}-validate-go-${{ hashFiles('**/go.sum') }} | ||||
|           restore-keys: ${{ runner.os }}-validate-go- | ||||
|  | ||||
|       - name: Install golangci-lint ${{ env.GOLANGCI_LINT_VERSION }} | ||||
|         run: curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b $(go env GOPATH)/bin ${GOLANGCI_LINT_VERSION} | ||||
|  | ||||
|       - name: Install missspell ${{ env.MISSSPELL_VERSION }} | ||||
|         run: curl -sfL https://raw.githubusercontent.com/golangci/misspell/master/install-misspell.sh | sh -s -- -b $(go env GOPATH)/bin ${MISSSPELL_VERSION} | ||||
|  | ||||
|       - name: Avoid generating webui | ||||
|         run: touch webui/static/index.html | ||||
|  | ||||
|       - name: Validate | ||||
|         run: make validate | ||||
|  | ||||
|   validate-generate: | ||||
|     runs-on: ubuntu-20.04 | ||||
|  | ||||
|     defaults: | ||||
|       run: | ||||
|         working-directory: ${{ github.workspace }}/go/src/github.com/traefik/traefik | ||||
|  | ||||
|     steps: | ||||
|       - name: Set up Go ${{ env.GO_VERSION }} | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: ${{ env.GO_VERSION }} | ||||
|  | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           path: go/src/github.com/traefik/traefik | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Cache Go modules | ||||
|         uses: actions/cache@v3 | ||||
|         with: | ||||
|           path: | | ||||
|             ~/go/pkg/mod | ||||
|             ~/.cache/go-build | ||||
|           key: ${{ runner.os }}-validate-generate-go-${{ hashFiles('**/go.sum') }} | ||||
|           restore-keys: ${{ runner.os }}-validate-generate-go- | ||||
|  | ||||
|       - name: go generate | ||||
|         run: | | ||||
|           go generate | ||||
|           git diff --exit-code | ||||
|  | ||||
|       - name: go mod tidy | ||||
|         run: | | ||||
|           go mod tidy | ||||
|           git diff --exit-code | ||||
|  | ||||
|       - name: make generate-crd | ||||
|         run: | | ||||
|           make generate-crd | ||||
|           git diff --exit-code | ||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -7,7 +7,6 @@ | ||||
| /webui/.tmp/ | ||||
| /site/ | ||||
| /docs/site/ | ||||
| /static/ | ||||
| /autogen/ | ||||
| /traefik | ||||
| /traefik.toml | ||||
| @@ -16,3 +15,7 @@ | ||||
| *.exe | ||||
| cover.out | ||||
| vendor/ | ||||
| plugins-storage/ | ||||
| plugins-local/ | ||||
| traefik_changelog.md | ||||
| integration/tailscale.secret | ||||
|   | ||||
							
								
								
									
										115
									
								
								.golangci.toml
									
									
									
									
									
								
							
							
						
						
									
										115
									
								
								.golangci.toml
									
									
									
									
									
								
							| @@ -1,115 +0,0 @@ | ||||
| [run] | ||||
|   timeout = "10m" | ||||
|   skip-files = [] | ||||
|   skip-dirs = [ | ||||
|     "pkg/provider/kubernetes/crd/generated/", | ||||
|   ] | ||||
|  | ||||
| [linters-settings] | ||||
|  | ||||
|   [linters-settings.govet] | ||||
|     check-shadowing = false | ||||
|  | ||||
|   [linters-settings.golint] | ||||
|     min-confidence = 0.0 | ||||
|  | ||||
|   [linters-settings.gocyclo] | ||||
|     min-complexity = 14.0 | ||||
|  | ||||
|   [linters-settings.maligned] | ||||
|     suggest-new = true | ||||
|  | ||||
|   [linters-settings.goconst] | ||||
|     min-len = 3.0 | ||||
|     min-occurrences = 4.0 | ||||
|  | ||||
|   [linters-settings.misspell] | ||||
|     locale = "US" | ||||
|  | ||||
|   [linters-settings.funlen] | ||||
|     lines = 230 # default 60 | ||||
|     statements = 120 # default 40 | ||||
|  | ||||
| [linters] | ||||
|   enable-all = true | ||||
|   disable = [ | ||||
|     "gocyclo", # FIXME must be fixed | ||||
|     "gosec", | ||||
|     "dupl", | ||||
|     "maligned", | ||||
|     "lll", | ||||
|     "unparam", | ||||
|     "prealloc", | ||||
|     "scopelint", | ||||
|     "gochecknoinits", | ||||
|     "gochecknoglobals", | ||||
|     "godox", | ||||
|     "gocognit", | ||||
|     "bodyclose", # Too many false-positive and panics. | ||||
|     "wsl", # Too strict | ||||
|     "gomnd", # Too strict | ||||
|     "stylecheck", # skip because report issues related to some generated files. | ||||
|     "testpackage", # Too strict | ||||
|     "goerr113", # Too strict | ||||
|     "nestif", # Too many false-positive. | ||||
|     "noctx", # Too strict | ||||
|     "exhaustive", # Too strict | ||||
|   ] | ||||
|  | ||||
| [issues] | ||||
|   exclude-use-default = false | ||||
|   max-per-linter = 0 | ||||
|   max-same-issues = 0 | ||||
|   exclude = [ | ||||
|     "SA1019: http.CloseNotifier is deprecated: the CloseNotifier interface predates Go's context package. New code should use Request.Context instead.", # FIXME must be fixed | ||||
|     "Error return value of .((os\\.)?std(out|err)\\..*|.*Close|.*Flush|os\\.Remove(All)?|.*printf?|os\\.(Un)?Setenv). is not checked", | ||||
|     "should have a package comment, unless it's in another file for this package", | ||||
|   ] | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "(.+)_test.go" | ||||
|     linters = ["goconst", "funlen", "godot"] | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "integration/.+_test.go" | ||||
|     text = "Error return value of `cmd\\.Process\\.Kill` is not checked" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "integration/(consul_catalog_test|constraint_test).go" | ||||
|     text = "Error return value of `(s.deregisterService|s.deregisterAgentService)` is not checked" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "integration/grpc_test.go" | ||||
|     text = "Error return value of `closer` is not checked" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/h2c/h2c.go" | ||||
|     text = "Error return value of `rw.Write` is not checked" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/middlewares/recovery/recovery.go" | ||||
|     text = "`logger` can be `github.com/stretchr/testify/assert.TestingT`" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/provider/docker/builder_test.go" | ||||
|     text = "(U1000: func )?`(.+)` is unused" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/provider/kubernetes/builder_(endpoint|service)_test.go" | ||||
|     text = "(U1000: func )?`(.+)` is unused" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/config/parser/.+_test.go" | ||||
|     text = "U1000: field `(foo|fuu)` is unused" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/server/service/bufferpool.go" | ||||
|     text = "SA6002: argument should be pointer-like to avoid allocations" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "cmd/configuration.go" | ||||
|     text = "string `traefik` has (\\d) occurrences, make it a constant" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/server/middleware/middlewares.go" | ||||
|     text = "Function 'buildConstructor' is too long \\(\\d+ > 230\\)" | ||||
|  [[issues.exclude-rules]] # FIXME must be fixed | ||||
|     path = "cmd/context.go" | ||||
|     text = "S1000: should use a simple channel send/receive instead of `select` with a single case" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/tracing/haystack/logger.go" | ||||
|     linters = ["goprintffuncname"] | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/tracing/tracing.go" | ||||
|     text = "printf-like formatting function 'SetErrorWithEvent' should be named 'SetErrorWithEventf'" | ||||
|  [[issues.exclude-rules]] | ||||
|     path = "pkg/log/deprecated.go" | ||||
|     linters = ["godot"] | ||||
							
								
								
									
										284
									
								
								.golangci.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										284
									
								
								.golangci.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,284 @@ | ||||
| run: | ||||
|   timeout: 10m | ||||
|   skip-files: [] | ||||
|   skip-dirs: | ||||
|     - pkg/provider/kubernetes/crd/generated/ | ||||
|  | ||||
| linters-settings: | ||||
|   govet: | ||||
|     enable-all: true | ||||
|     disable: | ||||
|       - shadow | ||||
|       - fieldalignment | ||||
|   gocyclo: | ||||
|     min-complexity: 14 | ||||
|   goconst: | ||||
|     min-len: 3 | ||||
|     min-occurrences: 4 | ||||
|   misspell: | ||||
|     locale: US | ||||
|   funlen: | ||||
|     lines: -1 | ||||
|     statements: 120 | ||||
|   forbidigo: | ||||
|     forbid: | ||||
|       - ^print(ln)?$ | ||||
|       - ^spew\.Print(f|ln)?$ | ||||
|       - ^spew\.Dump$ | ||||
|   depguard: | ||||
|     rules: | ||||
|       main: | ||||
|         deny: | ||||
|           - pkg: "github.com/instana/testify" | ||||
|             desc: not allowed | ||||
|           - pkg: "github.com/pkg/errors" | ||||
|             desc: Should be replaced by standard lib errors package | ||||
|   godox: | ||||
|     keywords: | ||||
|       - FIXME | ||||
|   importas: | ||||
|     no-unaliased: true | ||||
|     alias: | ||||
|       - alias: composeapi | ||||
|         pkg: github.com/docker/compose/v2/pkg/api | ||||
|  | ||||
|       # Standard Kubernetes rewrites: | ||||
|       - alias: corev1 | ||||
|         pkg: "k8s.io/api/core/v1" | ||||
|       - alias: netv1 | ||||
|         pkg: "k8s.io/api/networking/v1" | ||||
|       - alias: netv1beta1 | ||||
|         pkg: "k8s.io/api/networking/v1beta1" | ||||
|       - alias: admv1 | ||||
|         pkg: "k8s.io/api/admission/v1" | ||||
|       - alias: admv1beta1 | ||||
|         pkg: "k8s.io/api/admission/v1beta1" | ||||
|       - alias: extv1beta1 | ||||
|         pkg: "k8s.io/api/extensions/v1beta1" | ||||
|       - alias: metav1 | ||||
|         pkg: "k8s.io/apimachinery/pkg/apis/meta/v1" | ||||
|       - alias: ktypes | ||||
|         pkg: "k8s.io/apimachinery/pkg/types" | ||||
|       - alias: kerror | ||||
|         pkg: "k8s.io/apimachinery/pkg/api/errors" | ||||
|       - alias: kclientset | ||||
|         pkg: "k8s.io/client-go/kubernetes" | ||||
|       - alias: kinformers | ||||
|         pkg: "k8s.io/client-go/informers" | ||||
|       - alias: ktesting | ||||
|         pkg: "k8s.io/client-go/testing" | ||||
|       - alias: kschema | ||||
|         pkg: "k8s.io/apimachinery/pkg/runtime/schema" | ||||
|       - alias: kscheme | ||||
|         pkg: "k8s.io/client-go/kubernetes/scheme" | ||||
|       - alias: kversion | ||||
|         pkg: "k8s.io/apimachinery/pkg/version" | ||||
|       - alias: kubefake | ||||
|         pkg: "k8s.io/client-go/kubernetes/fake" | ||||
|       - alias: discoveryfake | ||||
|         pkg: "k8s.io/client-go/discovery/fake" | ||||
|  | ||||
|       # Kubernetes Gateway rewrites: | ||||
|       - alias: gateclientset | ||||
|         pkg: "sigs.k8s.io/gateway-api/pkg/client/clientset/gateway/versioned" | ||||
|       - alias: gateinformers | ||||
|         pkg: "sigs.k8s.io/gateway-api/pkg/client/informers/gateway/externalversions" | ||||
|       - alias: gatev1alpha2 | ||||
|         pkg: "sigs.k8s.io/gateway-api/apis/v1alpha2" | ||||
|  | ||||
|       # Traefik Kubernetes rewrites: | ||||
|       - alias: containousv1alpha1 | ||||
|         pkg: "github.com/traefik/traefik/v2/pkg/provider/kubernetes/crd/traefikcontainous/v1alpha1" | ||||
|       - alias: traefikv1alpha1 | ||||
|         pkg: "github.com/traefik/traefik/v2/pkg/provider/kubernetes/crd/traefikio/v1alpha1" | ||||
|       - alias: traefikclientset | ||||
|         pkg: "github.com/traefik/traefik/v2/pkg/provider/kubernetes/crd/generated/clientset/versioned" | ||||
|       - alias: traefikinformers | ||||
|         pkg: "github.com/traefik/traefik/v2/pkg/provider/kubernetes/crd/generated/informers/externalversions" | ||||
|       - alias: traefikscheme | ||||
|         pkg: "github.com/traefik/traefik/v2/pkg/provider/kubernetes/crd/generated/clientset/versioned/scheme" | ||||
|       - alias: traefikcrdfake | ||||
|         pkg: "github.com/traefik/traefik/v2/pkg/provider/kubernetes/crd/generated/clientset/versioned/fake" | ||||
|   tagalign: | ||||
|     align: false | ||||
|     sort: true | ||||
|     order: | ||||
|       - description | ||||
|       - json | ||||
|       - toml | ||||
|       - yaml | ||||
|       - yml | ||||
|       - label | ||||
|       - label-slice-as-struct | ||||
|       - file | ||||
|       - kv | ||||
|       - export | ||||
|   revive: | ||||
|     rules: | ||||
|       - name: struct-tag | ||||
|       - name: blank-imports | ||||
|       - name: context-as-argument | ||||
|       - name: context-keys-type | ||||
|       - name: dot-imports | ||||
|       - name: error-return | ||||
|       - name: error-strings | ||||
|       - name: error-naming | ||||
|       - name: exported | ||||
|         disabled: true | ||||
|       - name: if-return | ||||
|       - name: increment-decrement | ||||
|       - name: var-naming | ||||
|       - name: var-declaration | ||||
|       - name: package-comments | ||||
|         disabled: true | ||||
|       - name: range | ||||
|       - name: receiver-naming | ||||
|       - name: time-naming | ||||
|       - name: unexported-return | ||||
|       - name: indent-error-flow | ||||
|       - name: errorf | ||||
|       - name: empty-block | ||||
|       - name: superfluous-else | ||||
|       - name: unused-parameter | ||||
|         disabled: true | ||||
|       - name: unreachable-code | ||||
|       - name: redefines-builtin-id | ||||
|   gomoddirectives: | ||||
|     replace-allow-list: | ||||
|       - github.com/abbot/go-http-auth | ||||
|       - github.com/go-check/check | ||||
|       - github.com/gorilla/mux | ||||
|       - github.com/mailgun/minheap | ||||
|       - github.com/mailgun/multibuf | ||||
|       - github.com/jaguilar/vt100 | ||||
|   testifylint: | ||||
|     enable: | ||||
|       - bool-compare | ||||
|       - compares | ||||
|       - empty | ||||
|       - error-is-as | ||||
|       - error-nil | ||||
|       - expected-actual | ||||
|       - float-compare | ||||
|       - len | ||||
|       - suite-dont-use-pkg | ||||
|       - suite-extra-assert-call | ||||
|       - suite-thelper | ||||
|  | ||||
| linters: | ||||
|   enable-all: true | ||||
|   disable: | ||||
|     - deadcode # deprecated | ||||
|     - exhaustivestruct # deprecated | ||||
|     - golint # deprecated | ||||
|     - ifshort # deprecated | ||||
|     - interfacer # deprecated | ||||
|     - maligned # deprecated | ||||
|     - nosnakecase # deprecated | ||||
|     - scopelint # deprecated | ||||
|     - scopelint # deprecated | ||||
|     - structcheck # deprecated | ||||
|     - varcheck # deprecated | ||||
|     - sqlclosecheck # not relevant (SQL) | ||||
|     - rowserrcheck # not relevant (SQL) | ||||
|     - execinquery # not relevant (SQL) | ||||
|     - cyclop # duplicate of gocyclo | ||||
|     - lll # Not relevant | ||||
|     - gocyclo # FIXME must be fixed | ||||
|     - gocognit # Too strict | ||||
|     - nestif # Too many false-positive. | ||||
|     - prealloc # Too many false-positive. | ||||
|     - makezero # Not relevant | ||||
|     - dupl # Too strict | ||||
|     - gosec # Too strict | ||||
|     - gochecknoinits | ||||
|     - gochecknoglobals | ||||
|     - wsl # Too strict | ||||
|     - nlreturn # Not relevant | ||||
|     - gomnd # Too strict | ||||
|     - stylecheck # skip because report issues related to some generated files. | ||||
|     - testpackage # Too strict | ||||
|     - tparallel # Not relevant | ||||
|     - paralleltest # Not relevant | ||||
|     - exhaustive # Not relevant | ||||
|     - exhaustruct # Not relevant | ||||
|     - goerr113 # Too strict | ||||
|     - wrapcheck # Too strict | ||||
|     - noctx # Too strict | ||||
|     - bodyclose # too many false-positive | ||||
|     - forcetypeassert # Too strict | ||||
|     - tagliatelle # Too strict | ||||
|     - varnamelen # Not relevant | ||||
|     - nilnil # Not relevant | ||||
|     - ireturn # Not relevant | ||||
|     - contextcheck # too many false-positive | ||||
|     - containedctx # too many false-positive | ||||
|     - maintidx # kind of duplicate of gocyclo | ||||
|     - nonamedreturns # Too strict | ||||
|     - gosmopolitan  # not relevant | ||||
|  | ||||
| issues: | ||||
|   exclude-use-default: false | ||||
|   max-per-linter: 0 | ||||
|   max-same-issues: 0 | ||||
|   exclude: | ||||
|     - 'Error return value of .((os\.)?std(out|err)\..*|.*Close|.*Flush|os\.Remove(All)?|.*printf?|os\.(Un)?Setenv). is not checked' | ||||
|     - "should have a package comment, unless it's in another file for this package" | ||||
|     - 'SA1019: http.CloseNotifier has been deprecated' # FIXME must be fixed | ||||
|     - 'SA1019: cfg.SSLRedirect is deprecated' | ||||
|     - 'SA1019: cfg.SSLTemporaryRedirect is deprecated' | ||||
|     - 'SA1019: cfg.SSLHost is deprecated' | ||||
|     - 'SA1019: cfg.SSLForceHost is deprecated' | ||||
|     - 'SA1019: cfg.FeaturePolicy is deprecated' | ||||
|     - 'SA1019: c.Providers.ConsulCatalog.Namespace is deprecated' | ||||
|     - 'SA1019: c.Providers.Consul.Namespace is deprecated' | ||||
|     - 'SA1019: c.Providers.Nomad.Namespace is deprecated' | ||||
|   exclude-rules: | ||||
|     - path: '(.+)_test.go' | ||||
|       linters: | ||||
|         - goconst | ||||
|         - funlen | ||||
|         - godot | ||||
|     - path: '(.+)_test.go' | ||||
|       text: ' always receives ' | ||||
|       linters: | ||||
|         - unparam | ||||
|     - path: '(.+)\.go' | ||||
|       text: 'struct-tag: unknown option ''inline'' in JSON tag' | ||||
|       linters: | ||||
|         - revive | ||||
|     - path: pkg/server/service/bufferpool.go | ||||
|       text: 'SA6002: argument should be pointer-like to avoid allocations' | ||||
|     - path: pkg/server/middleware/middlewares.go | ||||
|       text: "Function 'buildConstructor' has too many statements" | ||||
|       linters: | ||||
|         - funlen | ||||
|     - path: pkg/tracing/haystack/logger.go | ||||
|       linters: | ||||
|         - goprintffuncname | ||||
|     - path: pkg/tracing/tracing.go | ||||
|       text: "printf-like formatting function 'SetErrorWithEvent' should be named 'SetErrorWithEventf'" | ||||
|       linters: | ||||
|         - goprintffuncname | ||||
|     - path: pkg/tls/tlsmanager_test.go | ||||
|       text: 'SA1019: config.ClientCAs.Subjects has been deprecated since Go 1.18' | ||||
|     - path: pkg/types/tls_test.go | ||||
|       text: 'SA1019: tlsConfig.RootCAs.Subjects has been deprecated since Go 1.18' | ||||
|     - path: pkg/provider/kubernetes/(crd|gateway)/client.go | ||||
|       linters: | ||||
|         - interfacebloat | ||||
|     - path: pkg/metrics/metrics.go | ||||
|       linters: | ||||
|         - interfacebloat | ||||
|     - path: integration/healthcheck_test.go | ||||
|       text: 'Duplicate words \(wsp2,\) found' | ||||
|       linters: | ||||
|         - dupword | ||||
|     - path: pkg/types/domain_test.go | ||||
|       text: 'Duplicate words \(sub\) found' | ||||
|       linters: | ||||
|         - dupword | ||||
|     - path: pkg/provider/kubernetes/gateway/client_mock_test.go | ||||
|       text: 'unusedwrite: unused write to field' | ||||
|       linters: | ||||
|         - govet | ||||
| @@ -1,43 +1,49 @@ | ||||
| project_name: traefik | ||||
| 
 | ||||
| dist: "./dist/[[ .GOOS ]]" | ||||
| 
 | ||||
| [[ if eq .GOOS "linux" ]] | ||||
| before: | ||||
|   hooks: | ||||
|     - go generate | ||||
| [[ end ]] | ||||
| 
 | ||||
| builds: | ||||
|   - binary: traefik | ||||
| 
 | ||||
|     main: ./cmd/traefik/traefik.go | ||||
|     main: ./cmd/traefik/ | ||||
|     env: | ||||
|       - CGO_ENABLED=0 | ||||
|     ldflags: | ||||
|       - -s -w -X github.com/containous/traefik/v2/pkg/version.Version={{.Version}} -X github.com/containous/traefik/v2/pkg/version.Codename={{.Env.CODENAME}} -X github.com/containous/traefik/v2/pkg/version.BuildDate={{.Date}} | ||||
| 
 | ||||
|       - -s -w -X github.com/traefik/traefik/v2/pkg/version.Version={{.Version}} -X github.com/traefik/traefik/v2/pkg/version.Codename={{.Env.CODENAME}} -X github.com/traefik/traefik/v2/pkg/version.BuildDate={{.Date}} | ||||
|     flags: | ||||
|       - -trimpath | ||||
|     goos: | ||||
|       - linux | ||||
|       - darwin | ||||
|       - windows | ||||
|       - freebsd | ||||
|       - openbsd | ||||
|       - "[[ .GOOS ]]" | ||||
|     goarch: | ||||
|       - amd64 | ||||
|       - 386 | ||||
|       - '386' | ||||
|       - arm | ||||
|       - arm64 | ||||
|       - ppc64le | ||||
|       - s390x | ||||
|       - riscv64 | ||||
|     goarm: | ||||
|       - 7 | ||||
|       - 6 | ||||
|       - 5 | ||||
|       - '7' | ||||
|       - '6' | ||||
|     ignore: | ||||
|       - goos: darwin | ||||
|         goarch: 386 | ||||
|         goarch: '386' | ||||
|       - goos: openbsd | ||||
|         goarch: arm | ||||
|       - goos: openbsd | ||||
|         goarch: arm64 | ||||
|       - goos: freebsd | ||||
|         goarch: arm | ||||
|       - goos: freebsd | ||||
|         goarch: arm64 | ||||
|       - goos: windows | ||||
|         goarch: arm | ||||
| 
 | ||||
| changelog: | ||||
|   skip: true | ||||
							
								
								
									
										83
									
								
								.semaphore/semaphore.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								.semaphore/semaphore.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,83 @@ | ||||
| version: v1.0 | ||||
| name: Traefik | ||||
| agent: | ||||
|   machine: | ||||
|     type: e1-standard-4 | ||||
|     os_image: ubuntu2004 | ||||
|  | ||||
| fail_fast: | ||||
|   stop: | ||||
|     when: "branch != 'master'" | ||||
|  | ||||
| auto_cancel: | ||||
|   queued: | ||||
|     when: "branch != 'master'" | ||||
|   running: | ||||
|     when: "branch != 'master'" | ||||
|  | ||||
| global_job_config: | ||||
|   prologue: | ||||
|     commands: | ||||
|       - curl -sSfL https://raw.githubusercontent.com/ldez/semgo/master/godownloader.sh | sudo sh -s -- -b "/usr/local/bin" | ||||
|       - sudo semgo go1.21 | ||||
|       - export "GOPATH=$(go env GOPATH)" | ||||
|       - export "SEMAPHORE_GIT_DIR=${GOPATH}/src/github.com/traefik/${SEMAPHORE_PROJECT_NAME}" | ||||
|       - export "PATH=${GOPATH}/bin:${PATH}" | ||||
|       - mkdir -vp "${SEMAPHORE_GIT_DIR}" "${GOPATH}/bin" | ||||
|       - export GOPROXY=https://proxy.golang.org,direct | ||||
|       - curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b "${GOPATH}/bin" v1.55.2 | ||||
|       - curl -sSfL https://gist.githubusercontent.com/traefiker/6d7ac019c11d011e4f131bb2cca8900e/raw/goreleaser.sh | bash -s -- -b "${GOPATH}/bin" | ||||
|       - checkout | ||||
|       - cache restore traefik-$(checksum go.sum) | ||||
|  | ||||
| blocks: | ||||
|   - name: Test Integration | ||||
|     dependencies: [] | ||||
|     run: | ||||
|       when: "branch =~ '.*' OR pull_request =~'.*'" | ||||
|     task: | ||||
|       jobs: | ||||
|         - name: Test Integration | ||||
|           commands: | ||||
|             - make pull-images | ||||
|             - touch webui/static/index.html # Avoid generating webui | ||||
|             - IN_DOCKER="" make binary | ||||
|             - make test-integration | ||||
|             - df -h | ||||
|       epilogue: | ||||
|         always: | ||||
|           commands: | ||||
|             - cache store traefik-$(checksum go.sum) $HOME/go/pkg/mod | ||||
|  | ||||
|   - name: Release | ||||
|     dependencies: [] | ||||
|     run: | ||||
|       when: "tag =~ '.*'" | ||||
|     task: | ||||
|       agent: | ||||
|         machine: | ||||
|           type: e1-standard-8 | ||||
|           os_image: ubuntu2004 | ||||
|       secrets: | ||||
|         - name: traefik | ||||
|       env_vars: | ||||
|         - name: GH_VERSION | ||||
|           value: 2.32.1 | ||||
|         - name: CODENAME | ||||
|           value: "mimolette" | ||||
|         - name: IN_DOCKER | ||||
|           value: "" | ||||
|       prologue: | ||||
|         commands: | ||||
|           - export VERSION=${SEMAPHORE_GIT_TAG_NAME} | ||||
|           - curl -sSL -o /tmp/gh_${GH_VERSION}_linux_amd64.tar.gz https://github.com/cli/cli/releases/download/v${GH_VERSION}/gh_${GH_VERSION}_linux_amd64.tar.gz | ||||
|           - tar -zxvf /tmp/gh_${GH_VERSION}_linux_amd64.tar.gz -C /tmp | ||||
|           - sudo mv /tmp/gh_${GH_VERSION}_linux_amd64/bin/gh /usr/local/bin/gh | ||||
|           - sudo rm -rf ~/.phpbrew ~/.kerl ~/.sbt ~/.nvm ~/.npm ~/.kiex /usr/lib/jvm /opt/az /opt/firefox /usr/lib/google-cloud-sdk ~/.rbenv ~/.pip_download_cache # Remove unnecessary data. | ||||
|           - sudo service docker stop && sudo umount /var/lib/docker && sudo service docker start # Unmounts the docker disk and the whole system disk is usable. | ||||
|       jobs: | ||||
|         - name: Release | ||||
|           commands: | ||||
|             - make release-packages | ||||
|             - gh release create ${SEMAPHORE_GIT_TAG_NAME} ./dist/**/traefik*.{zip,tar.gz} ./dist/traefik*.{tar.gz,txt} --repo traefik/traefik --title ${SEMAPHORE_GIT_TAG_NAME} --notes ${SEMAPHORE_GIT_TAG_NAME} | ||||
|             - ./script/deploy.sh | ||||
| @@ -1,4 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
| set -e | ||||
|  | ||||
| sudo rm -rf static | ||||
| @@ -1,20 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| set -e | ||||
|  | ||||
| curl -O https://dl.google.com/go/go"${GO_VERSION}".linux-amd64.tar.gz | ||||
|  | ||||
| tar -xvf go"${GO_VERSION}".linux-amd64.tar.gz | ||||
| rm -rf go"${GO_VERSION}".linux-amd64.tar.gz | ||||
|  | ||||
| sudo mkdir -p /usr/local/golang/"${GO_VERSION}"/go | ||||
| sudo mv go /usr/local/golang/"${GO_VERSION}"/ | ||||
|  | ||||
| sudo rm /usr/local/bin/go | ||||
| sudo chmod +x /usr/local/golang/"${GO_VERSION}"/go/bin/go | ||||
| sudo ln -s /usr/local/golang/"${GO_VERSION}"/go/bin/go /usr/local/bin/go | ||||
|  | ||||
| export GOROOT="/usr/local/golang/${GO_VERSION}/go" | ||||
| export GOTOOLDIR="/usr/local/golang/${GO_VERSION}/go/pkg/tool/linux_amd64" | ||||
|  | ||||
| go version | ||||
| @@ -1,6 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
| set -e | ||||
|  | ||||
| if [ -n "$SHOULD_TEST" ]; then ci_retry make pull-images; fi | ||||
|  | ||||
| if [ -n "$SHOULD_TEST" ]; then ci_retry make test-integration; fi | ||||
| @@ -1,8 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
| set -e | ||||
|  | ||||
| ci_retry make validate | ||||
|  | ||||
| if [ -n "$SHOULD_TEST" ]; then ci_retry make test-unit; fi | ||||
|  | ||||
| if [ -n "$SHOULD_TEST" ]; then make -j"${N_MAKE_JOBS}" crossbinary-default-parallel; fi | ||||
| @@ -1,35 +0,0 @@ | ||||
| # For personnal CI | ||||
| # mv /home/runner/workspace/src/github.com/<username>/ /home/runner/workspace/src/github.com/containous/ | ||||
| # cd /home/runner/workspace/src/github.com/containous/traefik/ | ||||
| for s in apache2 cassandra elasticsearch memcached mysql mongod postgresql sphinxsearch rethinkdb rabbitmq-server redis-server; do sudo service $s stop; done | ||||
| sudo swapoff -a | ||||
| sudo dd if=/dev/zero of=/swapfile bs=1M count=3072 | ||||
| sudo mkswap /swapfile | ||||
| sudo swapon /swapfile | ||||
| sudo rm -rf /home/runner/.rbenv | ||||
| sudo rm -rf /usr/local/golang/{1.4.3,1.5.4,1.6.4,1.7.6,1.8.6,1.9.7,1.10.3,1.11} | ||||
| #export DOCKER_VERSION=18.06.3 | ||||
| source .semaphoreci/vars | ||||
| if [ -z "${PULL_REQUEST_NUMBER}" ]; then SHOULD_TEST="-*-"; else TEMP_STORAGE=$(curl --silent https://patch-diff.githubusercontent.com/raw/containous/traefik/pull/${PULL_REQUEST_NUMBER}.diff | patch --dry-run -p1 -R || true); fi | ||||
| echo ${SHOULD_TEST} | ||||
| if [ -n "$TEMP_STORAGE" ]; then SHOULD_TEST=$(echo "$TEMP_STORAGE" | grep -Ev '(.md|.yaml|.yml)' || :); fi | ||||
| echo ${TEMP_STORAGE} | ||||
| echo ${SHOULD_TEST} | ||||
| #if [ -n "$SHOULD_TEST" ]; then sudo -E apt-get -yq update; fi | ||||
| #if [ -n "$SHOULD_TEST" ]; then sudo -E apt-get -yq --no-install-suggests --no-install-recommends --force-yes install docker-ce=${DOCKER_VERSION}*; fi | ||||
| if [ -n "$SHOULD_TEST" ]; then docker version; fi | ||||
| export GO_VERSION=1.13 | ||||
| if [ -f "./go.mod" ]; then GO_VERSION="$(grep '^go .*' go.mod | awk '{print $2}')"; export GO_VERSION; fi | ||||
| #if [ "${GO_VERSION}" == '1.14' ]; then export GO_VERSION=1.14rc2; fi | ||||
| echo "Selected Go version: ${GO_VERSION}" | ||||
|  | ||||
| if [ -f "./.semaphoreci/golang.sh" ]; then ./.semaphoreci/golang.sh; fi | ||||
| if [ -f "./.semaphoreci/golang.sh" ]; then export GOROOT="/usr/local/golang/${GO_VERSION}/go"; fi | ||||
| if [ -f "./.semaphoreci/golang.sh" ]; then export GOTOOLDIR="/usr/local/golang/${GO_VERSION}/go/pkg/tool/linux_amd64"; fi | ||||
| go version | ||||
|  | ||||
| if [ -f "./go.mod" ]; then export GO111MODULE=on; fi | ||||
| if [ -f "./go.mod" ]; then export GOPROXY=https://proxy.golang.org; fi | ||||
| if [ -f "./go.mod" ]; then go mod download; fi | ||||
|  | ||||
| df | ||||
| @@ -1,36 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
| set -e | ||||
|  | ||||
| export REPO='containous/traefik' | ||||
|  | ||||
| if VERSION=$(git describe --exact-match --abbrev=0 --tags); | ||||
| then | ||||
|   export VERSION | ||||
| else | ||||
|   export VERSION='' | ||||
| fi | ||||
|  | ||||
| export CODENAME=chevrotin | ||||
|  | ||||
| export N_MAKE_JOBS=2 | ||||
|  | ||||
|  | ||||
| function ci_retry { | ||||
|  | ||||
|     local NRETRY=3 | ||||
|     local NSLEEP=5 | ||||
|     local n=0 | ||||
|  | ||||
|     until [ $n -ge $NRETRY ] | ||||
|     do | ||||
|         "$@" && break | ||||
|         n=$((n+1)) | ||||
|         echo "${*} failed, attempt ${n}/${NRETRY}" | ||||
|         sleep $NSLEEP | ||||
|     done | ||||
|  | ||||
|     [ $n -lt $NRETRY ] | ||||
|  | ||||
| } | ||||
|  | ||||
| export -f ci_retry | ||||
							
								
								
									
										58
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										58
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,58 +0,0 @@ | ||||
| sudo: required | ||||
| dist: trusty | ||||
|  | ||||
| git: | ||||
|   depth: false | ||||
|  | ||||
| services: | ||||
|   - docker | ||||
|  | ||||
| env: | ||||
|   global: | ||||
|     - REPO=$TRAVIS_REPO_SLUG | ||||
|     - VERSION=$TRAVIS_TAG | ||||
|     - CODENAME=chevrotin | ||||
|     - GO111MODULE=on | ||||
|  | ||||
| script: | ||||
| - echo "Skipping tests... (Tests are executed on SemaphoreCI)" | ||||
| - if [ "$TRAVIS_PULL_REQUEST" != "false" ]; then make docs; fi | ||||
|  | ||||
| before_deploy: | ||||
|   - > | ||||
|     if ! [ "$BEFORE_DEPLOY_RUN" ]; then | ||||
|       export BEFORE_DEPLOY_RUN=1; | ||||
|       sudo -E apt-get -yq update; | ||||
|       sudo -E apt-get -yq --no-install-suggests --no-install-recommends --force-yes install docker-ce=${DOCKER_VERSION}*; | ||||
|       docker version; | ||||
|       make build-image; | ||||
|       if [ "$TRAVIS_TAG" ]; then | ||||
|         make release-packages; | ||||
|       fi; | ||||
|       curl -sfL https://raw.githubusercontent.com/containous/structor/master/godownloader.sh | bash -s -- -b "${GOPATH}/bin" ${STRUCTOR_VERSION} | ||||
|       structor -o containous -r traefik --dockerfile-url="https://raw.githubusercontent.com/containous/traefik/v1.7/docs.Dockerfile" --menu.js-url="https://raw.githubusercontent.com/containous/structor/master/traefik-menu.js.gotmpl" --rqts-url="https://raw.githubusercontent.com/containous/structor/master/requirements-override.txt" --force-edit-url --exp-branch=master --debug; | ||||
|     fi | ||||
|  | ||||
| deploy: | ||||
|   - provider: releases | ||||
|     api_key: ${GITHUB_TOKEN} | ||||
|     file: dist/traefik* | ||||
|     skip_cleanup: true | ||||
|     file_glob: true | ||||
|     on: | ||||
|       repo: containous/traefik | ||||
|       tags: true | ||||
|   - provider: script | ||||
|     script: sh script/deploy.sh | ||||
|     skip_cleanup: true | ||||
|     on: | ||||
|       repo: containous/traefik | ||||
|       tags: true | ||||
|   - provider: pages | ||||
|     edge: false | ||||
|     github_token: ${GITHUB_TOKEN} | ||||
|     local_dir: site | ||||
|     skip_cleanup: true | ||||
|     on: | ||||
|       repo: containous/traefik | ||||
|       all_branches: true | ||||
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										9783
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										9783
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| ## Our Pledge | ||||
|  | ||||
| In the interest of fostering an open and welcoming environment, we as contributors and maintainers pledge to making participation in our project and our community a harassment-free experience for everyone, regardless of age, body size, disability, ethnicity, gender identity and expression, level of experience,nationality, personal appearance, race, religion, or sexual identity and orientation. | ||||
| In the interest of fostering an open and welcoming environment, we as contributors and maintainers pledge to making participation in our project and our community a harassment-free experience for everyone, regardless of age, body size, disability, ethnicity, gender identity and expression, level of experience, nationality, personal appearance, race, religion, or sexual identity and orientation. | ||||
|  | ||||
| ## Our Standards | ||||
|  | ||||
| @@ -30,22 +30,38 @@ Project maintainers have the right and responsibility to remove, edit, or reject | ||||
|  | ||||
| ## Scope | ||||
|  | ||||
| This Code of Conduct applies both within project spaces and in public spaces when an individual is representing the project or its community.  | ||||
| Examples of representing a project or community include using an official project e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event.  | ||||
| This Code of Conduct applies both within project spaces and in public spaces when an individual is representing the project or our community. | ||||
|  | ||||
| Examples of representing a project or community include using an official project e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event. | ||||
| Representation of a project may be further defined and clarified by project maintainers. | ||||
|  | ||||
| ## Enforcement | ||||
|  | ||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by contacting the project team at contact@containo.us | ||||
| All complaints will be reviewed and investigated and will result in a response that is deemed necessary and appropriate to the circumstances.  | ||||
| The project team is obligated to maintain confidentiality with regard to the reporter of an incident.  | ||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by contacting the project team at contact@traefik.io | ||||
|  | ||||
| All complaints will be reviewed and investigated and will result in a response that is deemed necessary and appropriate to the circumstances. | ||||
|  | ||||
| The project team is obligated to maintain confidentiality with regard to the reporter of an incident. | ||||
|  | ||||
| Further details of specific enforcement policies may be posted separately. | ||||
|  | ||||
| Project maintainers who do not follow or enforce the Code of Conduct in good faith may face temporary or permanent repercussions as determined by other members of the project's leadership. | ||||
|  | ||||
| When an inapropriate behavior is reported, maintainers will discuss on the Maintainer's Discord before marking the message as "abuse".  | ||||
| This conversation beforehand avoids one-sided decisions. | ||||
|  | ||||
| The first message will be edited and marked as abuse. | ||||
| The second edited message and marked as abuse results in a 7-day ban. | ||||
| The third edited message and marked as abuse results in a permanent ban. | ||||
|  | ||||
| The content of edited messages is: | ||||
| `Dear user, we want traefik to provide a welcoming and respectful environment. Your [comment/issue/PR] has been reported and marked as abuse according to our [Code of Conduct](./CODE_OF_CONDUCT.md). Thank you.` | ||||
|  | ||||
| The [report must be resolved](https://docs.github.com/en/communities/moderating-comments-and-conversations/managing-reported-content-in-your-organizations-repository#resolving-a-report) accordingly. | ||||
|  | ||||
| ## Attribution | ||||
|  | ||||
| This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, available at [http://contributor-covenant.org/version/1/4][version] | ||||
|  | ||||
| [homepage]: http://contributor-covenant.org | ||||
| [version]: http://contributor-covenant.org/version/1/4/ | ||||
| [version]: http://contributor-covenant.org/version/1/4/ | ||||
|   | ||||
| @@ -1,4 +1,11 @@ | ||||
| # Contributing | ||||
|  | ||||
| - https://docs.traefik.io/contributing/submitting-pull-requests/ | ||||
| - https://docs.traefik.io/contributing/submitting-issues/ | ||||
| Here are some guidelines that should help to start contributing to the project. | ||||
|  | ||||
| - [Submitting pull Requests](https://doc.traefik.io/traefik/contributing/submitting-pull-requests/) | ||||
| - [Submitting issues](https://doc.traefik.io/traefik/contributing/submitting-issues/) | ||||
| - [Submitting security issues](https://doc.traefik.io/traefik/contributing/submitting-security-issues/) | ||||
| - [Advocating for Traefik](https://doc.traefik.io/traefik/contributing/advocating/) | ||||
| - [Triage Process](https://github.com/traefik/contributors-guide/blob/master/issue_triage.md) | ||||
|  | ||||
| If you are willing to become a maintainer of the project, please take a look at the [maintainers guidelines](docs/content/contributing/maintainers-guidelines.md). | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| The MIT License (MIT) | ||||
|  | ||||
| Copyright (c) 2016-2020 Containous SAS | ||||
| Copyright (c) 2016-2020 Containous SAS; 2020-2024 Traefik Labs | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
|   | ||||
							
								
								
									
										173
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										173
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,5 +1,3 @@ | ||||
| .PHONY: all docs docs-serve | ||||
|  | ||||
| SRCS = $(shell git ls-files '*.go' | grep -v '^vendor/') | ||||
|  | ||||
| TAG_NAME := $(shell git tag -l --contains HEAD) | ||||
| @@ -7,17 +5,16 @@ SHA := $(shell git rev-parse HEAD) | ||||
| VERSION_GIT := $(if $(TAG_NAME),$(TAG_NAME),$(SHA)) | ||||
| VERSION := $(if $(VERSION),$(VERSION),$(VERSION_GIT)) | ||||
|  | ||||
| BIND_DIR := "dist" | ||||
|  | ||||
| GIT_BRANCH := $(subst heads/,,$(shell git rev-parse --abbrev-ref HEAD 2>/dev/null)) | ||||
| TRAEFIK_DEV_IMAGE := traefik-dev$(if $(GIT_BRANCH),:$(subst /,-,$(GIT_BRANCH))) | ||||
|  | ||||
| REPONAME := $(shell echo $(REPO) | tr '[:upper:]' '[:lower:]') | ||||
| TRAEFIK_IMAGE := $(if $(REPONAME),$(REPONAME),"containous/traefik") | ||||
| TRAEFIK_IMAGE := $(if $(REPONAME),$(REPONAME),"traefik/traefik") | ||||
|  | ||||
| INTEGRATION_OPTS := $(if $(MAKE_DOCKER_HOST),-e "DOCKER_HOST=$(MAKE_DOCKER_HOST)", -e "TEST_CONTAINER=1" -v "/var/run/docker.sock:/var/run/docker.sock") | ||||
| INTEGRATION_OPTS := $(if $(MAKE_DOCKER_HOST),-e "DOCKER_HOST=$(MAKE_DOCKER_HOST)",-v "/var/run/docker.sock:/var/run/docker.sock") | ||||
| DOCKER_BUILD_ARGS := $(if $(DOCKER_VERSION), "--build-arg=DOCKER_VERSION=$(DOCKER_VERSION)",) | ||||
|  | ||||
| # only used when running in docker | ||||
| TRAEFIK_ENVS := \ | ||||
| 	-e OS_ARCH_ARG \ | ||||
| 	-e OS_PLATFORM_ARG \ | ||||
| @@ -27,125 +24,193 @@ TRAEFIK_ENVS := \ | ||||
| 	-e CODENAME \ | ||||
| 	-e TESTDIRS \ | ||||
| 	-e CI \ | ||||
| 	-e CONTAINER=DOCKER		# Indicator for integration tests that we are running inside a container. | ||||
| 	-e IN_DOCKER=true		# Indicator for integration tests that we are running inside a container. | ||||
|  | ||||
| TRAEFIK_MOUNT := -v "$(CURDIR)/$(BIND_DIR):/go/src/github.com/containous/traefik/$(BIND_DIR)" | ||||
| TRAEFIK_MOUNT := -v "$(CURDIR)/dist:/go/src/github.com/traefik/traefik/dist" | ||||
| DOCKER_RUN_OPTS := $(TRAEFIK_ENVS) $(TRAEFIK_MOUNT) "$(TRAEFIK_DEV_IMAGE)" | ||||
| DOCKER_NON_INTERACTIVE ?= false | ||||
| DOCKER_RUN_TRAEFIK := docker run --add-host=host.docker.internal:127.0.0.1 $(INTEGRATION_OPTS) $(if $(DOCKER_NON_INTERACTIVE), , -it) $(DOCKER_RUN_OPTS) | ||||
| DOCKER_RUN_TRAEFIK := docker run $(INTEGRATION_OPTS) $(if $(DOCKER_NON_INTERACTIVE), , -it) $(DOCKER_RUN_OPTS) | ||||
| DOCKER_RUN_TRAEFIK_TEST := docker run --add-host=host.docker.internal:127.0.0.1 --rm --name=traefik --network traefik-test-network -v $(PWD):$(PWD) -w $(PWD) $(INTEGRATION_OPTS) $(if $(DOCKER_NON_INTERACTIVE), , -it) $(DOCKER_RUN_OPTS) | ||||
| DOCKER_RUN_TRAEFIK_NOTTY := docker run $(INTEGRATION_OPTS) $(if $(DOCKER_NON_INTERACTIVE), , -i) $(DOCKER_RUN_OPTS) | ||||
|  | ||||
| PRE_TARGET ?= build-dev-image | ||||
| IN_DOCKER ?= true | ||||
|  | ||||
| .PHONY: default | ||||
| default: binary | ||||
|  | ||||
| ## Build Dev Docker image | ||||
| build-dev-image: dist | ||||
| 	docker build $(DOCKER_BUILD_ARGS) -t "$(TRAEFIK_DEV_IMAGE)" -f build.Dockerfile . | ||||
|  | ||||
| ## Build Dev Docker image without cache | ||||
| build-dev-image-no-cache: dist | ||||
| 	docker build --no-cache -t "$(TRAEFIK_DEV_IMAGE)" -f build.Dockerfile . | ||||
|  | ||||
| ## Create the "dist" directory | ||||
| dist: | ||||
| 	mkdir dist | ||||
| 	mkdir -p dist | ||||
|  | ||||
| ## Build Dev Docker image | ||||
| .PHONY: build-dev-image | ||||
| build-dev-image: dist | ||||
| ifneq ("$(IN_DOCKER)", "") | ||||
| 	docker build $(DOCKER_BUILD_ARGS) -t "$(TRAEFIK_DEV_IMAGE)" --build-arg HOST_PWD="$(PWD)" -f build.Dockerfile . | ||||
| endif | ||||
|  | ||||
| ## Build Dev Docker image without cache | ||||
| .PHONY: build-dev-image-no-cache | ||||
| build-dev-image-no-cache: dist | ||||
| ifneq ("$(IN_DOCKER)", "") | ||||
| 	docker build $(DOCKER_BUILD_ARGS) --no-cache -t "$(TRAEFIK_DEV_IMAGE)" --build-arg HOST_PWD="$(PWD)" -f build.Dockerfile . | ||||
| endif | ||||
|  | ||||
| ## Build WebUI Docker image | ||||
| .PHONY: build-webui-image | ||||
| build-webui-image: | ||||
| 	docker build -t traefik-webui -f webui/Dockerfile webui | ||||
|  | ||||
| ## Clean WebUI static generated assets | ||||
| .PHONY: clean-webui | ||||
| clean-webui: | ||||
| 	rm -r webui/static | ||||
| 	mkdir -p webui/static | ||||
| 	printf 'For more information see `webui/readme.md`' > webui/static/DONT-EDIT-FILES-IN-THIS-DIRECTORY.md | ||||
|  | ||||
| ## Generate WebUI | ||||
| generate-webui: build-webui-image | ||||
| 	if [ ! -d "static" ]; then \ | ||||
| 		mkdir -p static; \ | ||||
| 		docker run --rm -v "$$PWD/static":'/src/static' traefik-webui npm run build:nc; \ | ||||
| 		docker run --rm -v "$$PWD/static":'/src/static' traefik-webui chown -R $(shell id -u):$(shell id -g) ../static; \ | ||||
| 		echo 'For more informations show `webui/readme.md`' > $$PWD/static/DONT-EDIT-FILES-IN-THIS-DIRECTORY.md; \ | ||||
| 	fi | ||||
| webui/static/index.html: | ||||
| 	$(MAKE) build-webui-image | ||||
| 	docker run --rm -v "$(PWD)/webui/static":'/src/webui/static' traefik-webui npm run build:nc | ||||
| 	docker run --rm -v "$(PWD)/webui/static":'/src/webui/static' traefik-webui chown -R $(shell id -u):$(shell id -g) ./static | ||||
|  | ||||
| ## Build the linux binary | ||||
| binary: generate-webui $(PRE_TARGET) | ||||
| 	$(if $(PRE_TARGET),$(DOCKER_RUN_TRAEFIK)) ./script/make.sh generate binary | ||||
| .PHONY: generate-webui | ||||
| generate-webui: webui/static/index.html | ||||
|  | ||||
| ## Build the binary for the standard plaforms (linux, darwin, windows) | ||||
| ## Build the binary | ||||
| .PHONY: binary | ||||
| binary: generate-webui build-dev-image | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK)) ./script/make.sh generate binary | ||||
|  | ||||
| ## Build the linux binary locally | ||||
| .PHONY: binary-debug | ||||
| binary-debug: generate-webui | ||||
| 	GOOS=linux ./script/make.sh binary | ||||
|  | ||||
| ## Build the binary for the standard platforms (linux, darwin, windows) | ||||
| .PHONY: crossbinary-default | ||||
| crossbinary-default: generate-webui build-dev-image | ||||
| 	$(DOCKER_RUN_TRAEFIK_NOTTY) ./script/make.sh generate crossbinary-default | ||||
|  | ||||
| ## Build the binary for the standard plaforms (linux, darwin, windows) in parallel | ||||
| ## Build the binary for the standard platforms (linux, darwin, windows) in parallel | ||||
| .PHONY: crossbinary-default-parallel | ||||
| crossbinary-default-parallel: | ||||
| 	$(MAKE) generate-webui | ||||
| 	$(MAKE) build-dev-image crossbinary-default | ||||
|  | ||||
| ## Run the unit and integration tests | ||||
| .PHONY: test | ||||
| test: build-dev-image | ||||
| 	$(DOCKER_RUN_TRAEFIK) ./script/make.sh generate test-unit binary test-integration | ||||
| 	-docker network create traefik-test-network --driver bridge --subnet 172.31.42.0/24 | ||||
| 	trap 'docker network rm traefik-test-network' EXIT; \ | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK_TEST)) ./script/make.sh generate test-unit binary test-integration | ||||
|  | ||||
| ## Run the unit tests | ||||
| test-unit: $(PRE_TARGET) | ||||
| 	$(if $(PRE_TARGET),$(DOCKER_RUN_TRAEFIK)) ./script/make.sh generate test-unit | ||||
|  | ||||
| ## Pull all images for integration tests | ||||
| pull-images: | ||||
| 	grep --no-filename -E '^\s+image:' ./integration/resources/compose/*.yml | awk '{print $$2}' | sort | uniq | xargs -P 6 -n 1 docker pull | ||||
| .PHONY: test-unit | ||||
| test-unit: build-dev-image | ||||
| 	-docker network create traefik-test-network --driver bridge --subnet 172.31.42.0/24 | ||||
| 	trap 'docker network rm traefik-test-network' EXIT; \ | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK_TEST)) ./script/make.sh generate test-unit | ||||
|  | ||||
| ## Run the integration tests | ||||
| test-integration: $(PRE_TARGET) | ||||
| 	$(if $(PRE_TARGET),$(DOCKER_RUN_TRAEFIK),TEST_CONTAINER=1) ./script/make.sh generate binary test-integration | ||||
| 	TEST_HOST=1 ./script/make.sh test-integration | ||||
| .PHONY: test-integration | ||||
| test-integration: build-dev-image | ||||
| 	-docker network create traefik-test-network --driver bridge --subnet 172.31.42.0/24 | ||||
| 	trap 'docker network rm traefik-test-network' EXIT; \ | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK_TEST)) ./script/make.sh generate binary test-integration | ||||
|  | ||||
| ## Pull all images for integration tests | ||||
| .PHONY: pull-images | ||||
| pull-images: | ||||
| 	grep --no-filename -E '^\s+image:' ./integration/resources/compose/*.yml \ | ||||
| 		| awk '{print $$2}' \ | ||||
| 		| sort \ | ||||
| 		| uniq \ | ||||
| 		| xargs -P 6 -n 1 docker pull | ||||
|  | ||||
| ## Validate code and docs | ||||
| validate-files: $(PRE_TARGET) | ||||
| 	$(if $(PRE_TARGET),$(DOCKER_RUN_TRAEFIK)) ./script/make.sh generate validate-lint validate-misspell | ||||
| .PHONY: validate-files | ||||
| validate-files: build-dev-image | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK)) ./script/make.sh generate validate-lint validate-misspell | ||||
| 	bash $(CURDIR)/script/validate-shell-script.sh | ||||
|  | ||||
| ## Validate code, docs, and vendor | ||||
| validate: $(PRE_TARGET) | ||||
| 	$(if $(PRE_TARGET),$(DOCKER_RUN_TRAEFIK)) ./script/make.sh generate validate-lint validate-misspell validate-vendor | ||||
| .PHONY: validate | ||||
| validate: build-dev-image | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK)) ./script/make.sh generate validate-lint validate-misspell validate-vendor | ||||
| 	bash $(CURDIR)/script/validate-shell-script.sh | ||||
|  | ||||
| ## Clean up static directory and build a Docker Traefik image | ||||
| build-image: binary | ||||
| 	rm -rf static | ||||
| .PHONY: build-image | ||||
| build-image: clean-webui binary | ||||
| 	docker build -t $(TRAEFIK_IMAGE) . | ||||
|  | ||||
| ## Build a Docker Traefik image | ||||
| ## Build a Docker Traefik image without re-building the webui | ||||
| .PHONY: build-image-dirty | ||||
| build-image-dirty: binary | ||||
| 	docker build -t $(TRAEFIK_IMAGE) . | ||||
|  | ||||
| ## Locally build traefik for linux, then shove it an alpine image, with basic tools. | ||||
| .PHONY: build-image-debug | ||||
| build-image-debug: binary-debug | ||||
| 	docker build -t $(TRAEFIK_IMAGE) -f debug.Dockerfile . | ||||
|  | ||||
| ## Start a shell inside the build env | ||||
| .PHONY: shell | ||||
| shell: build-dev-image | ||||
| 	$(DOCKER_RUN_TRAEFIK) /bin/bash | ||||
|  | ||||
| ## Build documentation site | ||||
| .PHONY: docs | ||||
| docs: | ||||
| 	make -C ./docs docs | ||||
|  | ||||
| ## Serve the documentation site localy | ||||
| ## Serve the documentation site locally | ||||
| .PHONY: docs-serve | ||||
| docs-serve: | ||||
| 	make -C ./docs docs-serve | ||||
|  | ||||
| ## Generate CRD clientset | ||||
| ## Pull image for doc building | ||||
| .PHONY: docs-pull-images | ||||
| docs-pull-images: | ||||
| 	make -C ./docs docs-pull-images | ||||
|  | ||||
| ## Generate CRD clientset and CRD manifests | ||||
| .PHONY: generate-crd | ||||
| generate-crd: | ||||
| 	./script/update-generated-crd-code.sh | ||||
| 	@$(CURDIR)/script/code-gen-docker.sh | ||||
|  | ||||
| ## Generate code from dynamic configuration https://github.com/traefik/genconf | ||||
| .PHONY: generate-genconf | ||||
| generate-genconf: | ||||
| 	go run ./cmd/internal/gen/ | ||||
|  | ||||
| ## Create packages for the release | ||||
| .PHONY: release-packages | ||||
| release-packages: generate-webui build-dev-image | ||||
| 	rm -rf dist | ||||
| 	$(DOCKER_RUN_TRAEFIK_NOTTY) goreleaser release --skip-publish --timeout="60m" | ||||
| 	$(DOCKER_RUN_TRAEFIK_NOTTY) tar cfz dist/traefik-${VERSION}.src.tar.gz \ | ||||
| 	@- $(foreach os, linux darwin windows freebsd openbsd, \ | ||||
|         $(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK_NOTTY)) goreleaser release --skip-publish -p 2 --timeout="90m" --config $(shell go run ./internal/release $(os)); \ | ||||
|         $(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK_NOTTY)) go clean -cache; \ | ||||
|     ) | ||||
|  | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK_NOTTY)) cat dist/**/*_checksums.txt >> dist/traefik_${VERSION}_checksums.txt | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK_NOTTY)) rm dist/**/*_checksums.txt | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK_NOTTY)) tar cfz dist/traefik-${VERSION}.src.tar.gz \ | ||||
| 		--exclude-vcs \ | ||||
| 		--exclude .idea \ | ||||
| 		--exclude .travis \ | ||||
| 		--exclude .semaphoreci \ | ||||
| 		--exclude .github \ | ||||
| 		--exclude dist . | ||||
| 	$(DOCKER_RUN_TRAEFIK_NOTTY) chown -R $(shell id -u):$(shell id -g) dist/ | ||||
| 	$(if $(IN_DOCKER),$(DOCKER_RUN_TRAEFIK_NOTTY)) chown -R $(shell id -u):$(shell id -g) dist/ | ||||
|  | ||||
| ## Format the Code | ||||
| .PHONY: fmt | ||||
| fmt: | ||||
| 	gofmt -s -l -w $(SRCS) | ||||
|  | ||||
| .PHONY: run-dev | ||||
| run-dev: | ||||
| 	go generate | ||||
| 	GO111MODULE=on go build ./cmd/traefik | ||||
|   | ||||
							
								
								
									
										62
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,17 +1,19 @@ | ||||
|  | ||||
| <p align="center"> | ||||
| <img src="docs/content/assets/img/traefik.logo.png" alt="Traefik" title="Traefik" /> | ||||
|     <picture> | ||||
|       <source media="(prefers-color-scheme: dark)" srcset="docs/content/assets/img/traefik.logo-dark.png"> | ||||
|       <source media="(prefers-color-scheme: light)" srcset="docs/content/assets/img/traefik.logo.png"> | ||||
|       <img alt="Traefik" title="Traefik" src="docs/content/assets/img/traefik.logo.png"> | ||||
|     </picture> | ||||
| </p> | ||||
|  | ||||
| [](https://semaphoreci.com/containous/traefik) | ||||
| [](https://docs.traefik.io) | ||||
| [](https://goreportcard.com/report/containous/traefik) | ||||
| [](https://microbadger.com/images/traefik) | ||||
| [](https://github.com/containous/traefik/blob/master/LICENSE.md) | ||||
| [](https://community.containo.us/) | ||||
| [](https://doc.traefik.io/traefik) | ||||
| [](https://goreportcard.com/report/traefik/traefik) | ||||
| [](https://github.com/traefik/traefik/blob/master/LICENSE.md) | ||||
| [](https://community.traefik.io/) | ||||
| [](https://twitter.com/intent/follow?screen_name=traefik) | ||||
|  | ||||
|  | ||||
| Traefik (pronounced _traffic_) is a modern HTTP reverse proxy and load balancer that makes deploying microservices easy. | ||||
| Traefik integrates with your existing infrastructure components ([Docker](https://www.docker.com/), [Swarm mode](https://docs.docker.com/engine/swarm/), [Kubernetes](https://kubernetes.io), [Marathon](https://mesosphere.github.io/marathon/), [Consul](https://www.consul.io/), [Etcd](https://coreos.com/etcd/), [Rancher](https://rancher.com), [Amazon ECS](https://aws.amazon.com/ecs), ...) and configures itself automatically and dynamically. | ||||
| Pointing Traefik at your orchestrator should be the _only_ configuration step you need. | ||||
| @@ -33,7 +35,7 @@ Pointing Traefik at your orchestrator should be the _only_ configuration step yo | ||||
|  | ||||
| --- | ||||
|  | ||||
| :warning: Please be aware that the old configurations for Traefik v1.x are NOT compatible with the v2.x config as of now. If you're running v2, please ensure you are using a [v2 configuration](https://docs.traefik.io/). | ||||
| :warning: Please be aware that the old configurations for Traefik v1.x are NOT compatible with the v2.x config as of now. If you're running v2, please ensure you are using a [v2 configuration](https://doc.traefik.io/traefik/). | ||||
|  | ||||
| ## Overview | ||||
|  | ||||
| @@ -64,20 +66,19 @@ _(But if you'd rather configure some of your routes manually, Traefik supports t | ||||
| - Keeps access logs (JSON, CLF) | ||||
| - Fast | ||||
| - Exposes a Rest API | ||||
| - Packaged as a single binary file (made with :heart: with go) and available as a [tiny](https://microbadger.com/images/traefik) [official](https://hub.docker.com/r/_/traefik/) docker image | ||||
|  | ||||
| - Packaged as a single binary file (made with :heart: with go) and available as an [official](https://hub.docker.com/r/_/traefik/) docker image | ||||
|  | ||||
| ## Supported Backends | ||||
|  | ||||
| - [Docker](https://docs.traefik.io/providers/docker/) / [Swarm mode](https://docs.traefik.io/providers/docker/) | ||||
| - [Kubernetes](https://docs.traefik.io/providers/kubernetes-crd/) | ||||
| - [Marathon](https://docs.traefik.io/providers/marathon/) | ||||
| - [Rancher](https://docs.traefik.io/providers/rancher/) (Metadata) | ||||
| - [File](https://docs.traefik.io/providers/file/) | ||||
| - [Docker](https://doc.traefik.io/traefik/providers/docker/) / [Swarm mode](https://doc.traefik.io/traefik/providers/docker/) | ||||
| - [Kubernetes](https://doc.traefik.io/traefik/providers/kubernetes-crd/) | ||||
| - [Marathon](https://doc.traefik.io/traefik/providers/marathon/) | ||||
| - [Rancher](https://doc.traefik.io/traefik/providers/rancher/) (Metadata) | ||||
| - [File](https://doc.traefik.io/traefik/providers/file/) | ||||
|  | ||||
| ## Quickstart | ||||
|  | ||||
| To get your hands on Traefik, you can use the [5-Minute Quickstart](https://docs.traefik.io/getting-started/quick-start/) in our documentation (you will need Docker). | ||||
| To get your hands on Traefik, you can use the [5-Minute Quickstart](https://doc.traefik.io/traefik/getting-started/quick-start/) in our documentation (you will need Docker). | ||||
|  | ||||
| ## Web UI | ||||
|  | ||||
| @@ -87,28 +88,27 @@ You can access the simple HTML frontend of Traefik. | ||||
|  | ||||
| ## Documentation | ||||
|  | ||||
| You can find the complete documentation of Traefik v2 at [https://docs.traefik.io](https://docs.traefik.io). | ||||
|  | ||||
| If you are using Traefik v1, you can find the complete documentation at [https://docs.traefik.io/v1.7/](https://docs.traefik.io/v1.7/). | ||||
| You can find the complete documentation of Traefik v2 at [https://doc.traefik.io/traefik/](https://doc.traefik.io/traefik/). | ||||
|  | ||||
| A collection of contributions around Traefik can be found at [https://awesome.traefik.io](https://awesome.traefik.io). | ||||
|  | ||||
| ## Support | ||||
|  | ||||
| To get community support, you can: | ||||
| - join the Traefik community forum: [](https://community.containo.us/) | ||||
|  | ||||
| If you need commercial support, please contact [Containo.us](https://containo.us) by mail: <mailto:support@containo.us>. | ||||
| - join the Traefik community forum: [](https://community.traefik.io/) | ||||
|  | ||||
| If you need commercial support, please contact [Traefik.io](https://traefik.io) by mail: <mailto:support@traefik.io>. | ||||
|  | ||||
| ## Download | ||||
|  | ||||
| - Grab the latest binary from the [releases](https://github.com/containous/traefik/releases) page and run it with the [sample configuration file](https://raw.githubusercontent.com/containous/traefik/master/traefik.sample.toml): | ||||
| - Grab the latest binary from the [releases](https://github.com/traefik/traefik/releases) page and run it with the [sample configuration file](https://raw.githubusercontent.com/traefik/traefik/master/traefik.sample.toml): | ||||
|  | ||||
| ```shell | ||||
| ./traefik --configFile=traefik.toml | ||||
| ``` | ||||
|  | ||||
| - Or use the official tiny Docker image and run it with the [sample configuration file](https://raw.githubusercontent.com/containous/traefik/master/traefik.sample.toml): | ||||
| - Or use the official tiny Docker image and run it with the [sample configuration file](https://raw.githubusercontent.com/traefik/traefik/master/traefik.sample.toml): | ||||
|  | ||||
| ```shell | ||||
| docker run -d -p 8080:8080 -p 80:80 -v $PWD/traefik.toml:/etc/traefik/traefik.toml traefik | ||||
| @@ -117,16 +117,18 @@ docker run -d -p 8080:8080 -p 80:80 -v $PWD/traefik.toml:/etc/traefik/traefik.to | ||||
| - Or get the sources: | ||||
|  | ||||
| ```shell | ||||
| git clone https://github.com/containous/traefik | ||||
| git clone https://github.com/traefik/traefik | ||||
| ``` | ||||
|  | ||||
| ## Introductory Videos | ||||
|  | ||||
| You can find high level and deep dive videos on [videos.containo.us](https://videos.containo.us). | ||||
| You can find high level and deep dive videos on [videos.traefik.io](https://videos.traefik.io). | ||||
|  | ||||
| ## Maintainers | ||||
|  | ||||
| [Information about process and maintainers](docs/content/contributing/maintainers.md) | ||||
| We are strongly promoting a philosophy of openness and sharing, and firmly standing against the elitist closed approach. Being part of the core team should be accessible to anyone who is motivated and want to be part of that journey! | ||||
| This [document](docs/content/contributing/maintainers-guidelines.md) describes how to be part of the [maintainers' team](docs/content/contributing/maintainers.md) as well as various responsibilities and guidelines for Traefik maintainers. | ||||
| You can also find more information on our process to review pull requests and manage issues [in this document](https://github.com/traefik/contributors-guide/blob/master/issue_triage.md). | ||||
|  | ||||
| ## Contributing | ||||
|  | ||||
| @@ -137,7 +139,7 @@ By participating in this project, you agree to abide by its terms. | ||||
|  | ||||
| ## Release Cycle | ||||
|  | ||||
| - We release a new version (e.g. 1.1.0, 1.2.0, 1.3.0) every other month. | ||||
| - We usually release 3/4 new versions (e.g. 1.1.0, 1.2.0, 1.3.0) per year. | ||||
| - Release Candidates are available before the release (e.g. 1.1.0-rc1, 1.1.0-rc2, 1.1.0-rc3, 1.1.0-rc4, before 1.1.0). | ||||
| - Bug-fixes (e.g. 1.1.1, 1.1.2, 1.2.1, 1.2.3) are released as needed (no additional features are delivered in those versions, bug-fixes only). | ||||
|  | ||||
| @@ -152,9 +154,9 @@ We use [Semantic Versioning](https://semver.org/). | ||||
|  | ||||
| ## Credits | ||||
|  | ||||
| Kudos to [Peka](http://peka.byethost11.com/photoblog/) for his awesome work on the logo . | ||||
| Kudos to [Peka](http://peka.byethost11.com/photoblog/) for his awesome work on the gopher's logo!. | ||||
|  | ||||
| Traefik's logo is licensed under the Creative Commons 3.0 Attributions license. | ||||
| The gopher's logo of Traefik is licensed under the Creative Commons 3.0 Attributions license. | ||||
|  | ||||
| Traefik's logo was inspired by the gopher stickers made by [Takuya Ueda](https://twitter.com/tenntenn). | ||||
| The gopher's logo of Traefik was inspired by the gopher stickers made by [Takuya Ueda](https://twitter.com/tenntenn). | ||||
| The original Go gopher was designed by [Renee French](https://reneefrench.blogspot.com/). | ||||
|   | ||||
							
								
								
									
										30
									
								
								SECURITY.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								SECURITY.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| # Security Policy | ||||
|  | ||||
| You can join our security mailing list to be aware of the latest announcements from our security team. | ||||
| You can subscribe sending a mail to security+subscribe@traefik.io or on [the online viewer](https://groups.google.com/a/traefik.io/forum/#!forum/security). | ||||
|  | ||||
| Reported vulnerabilities can be found on [cve.mitre.org](https://cve.mitre.org/cgi-bin/cvekey.cgi?keyword=traefik). | ||||
|  | ||||
| ## Supported Versions | ||||
|  | ||||
| - We usually release 3/4 new versions (e.g. 1.1.0, 1.2.0, 1.3.0) per year. | ||||
| - Release Candidates are available before the release (e.g. 1.1.0-rc1, 1.1.0-rc2, 1.1.0-rc3, 1.1.0-rc4, before 1.1.0). | ||||
| - Bug-fixes (e.g. 1.1.1, 1.1.2, 1.2.1, 1.2.3) are released as needed (no additional features are delivered in those versions, bug-fixes only). | ||||
|  | ||||
| Each version is supported until the next one is released (e.g. 1.1.x will be supported until 1.2.0 is out). | ||||
|  | ||||
| We use [Semantic Versioning](https://semver.org/). | ||||
|  | ||||
| | Version   | Supported          | | ||||
| |-----------|--------------------| | ||||
| | `2.2.x`   | :white_check_mark: | | ||||
| | `< 2.2.x` | :x:                | | ||||
| | `1.7.x`   | :white_check_mark: | | ||||
| | `< 1.7.x` | :x:                | | ||||
|  | ||||
| ## Reporting a Vulnerability | ||||
|  | ||||
| We want to keep Traefik safe for everyone. | ||||
| If you've discovered a security vulnerability in Traefik, | ||||
| we appreciate your help in disclosing it to us in a responsible manner, | ||||
| by creating a [security advisory](https://github.com/traefik/traefik/security/advisories). | ||||
| @@ -1,7 +1,6 @@ | ||||
| FROM golang:1.14-alpine | ||||
| FROM golang:1.21-alpine | ||||
|  | ||||
| RUN apk --update upgrade \ | ||||
|     && apk --no-cache --no-progress add git mercurial bash gcc musl-dev curl tar ca-certificates tzdata \ | ||||
| RUN apk --no-cache --no-progress add git mercurial bash gcc musl-dev curl tar ca-certificates tzdata \ | ||||
|     && update-ca-certificates \ | ||||
|     && rm -rf /var/cache/apk/* | ||||
|  | ||||
| @@ -13,25 +12,26 @@ RUN mkdir -p /usr/local/bin \ | ||||
|     && curl -fL https://download.docker.com/linux/static/stable/x86_64/docker-${DOCKER_VERSION}.tgz \ | ||||
|     | tar -xzC /usr/local/bin --transform 's#^.+/##x' | ||||
|  | ||||
| # Download go-bindata binary to bin folder in $GOPATH | ||||
| RUN mkdir -p /usr/local/bin \ | ||||
|     && curl -fsSL -o /usr/local/bin/go-bindata https://github.com/containous/go-bindata/releases/download/v1.0.0/go-bindata \ | ||||
|     && chmod +x /usr/local/bin/go-bindata | ||||
|  | ||||
| # Download golangci-lint binary to bin folder in $GOPATH | ||||
| RUN curl -sfL https://install.goreleaser.com/github.com/golangci/golangci-lint.sh | bash -s -- -b $GOPATH/bin v1.28.0 | ||||
| RUN curl -sfL https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | bash -s -- -b $GOPATH/bin v1.55.2 | ||||
|  | ||||
| # Download misspell binary to bin folder in $GOPATH | ||||
| RUN  curl -sfL https://raw.githubusercontent.com/client9/misspell/master/install-misspell.sh | bash -s -- -b $GOPATH/bin v0.3.4 | ||||
| RUN curl -sfL https://raw.githubusercontent.com/golangci/misspell/master/install-misspell.sh | bash -s -- -b $GOPATH/bin v0.4.0 | ||||
|  | ||||
| # Download goreleaser binary to bin folder in $GOPATH | ||||
| RUN curl -sfL https://install.goreleaser.com/github.com/goreleaser/goreleaser.sh | sh | ||||
| RUN curl -sfL https://gist.githubusercontent.com/traefiker/6d7ac019c11d011e4f131bb2cca8900e/raw/goreleaser.sh | sh | ||||
|  | ||||
| WORKDIR /go/src/github.com/containous/traefik | ||||
| WORKDIR /go/src/github.com/traefik/traefik | ||||
|  | ||||
| # Because of CVE-2022-24765 (https://github.blog/2022-04-12-git-security-vulnerability-announced/), | ||||
| # we configure git to allow the Traefik codebase path on the Host for docker in docker usages. | ||||
| ARG HOST_PWD="" | ||||
|  | ||||
| RUN git config --global --add safe.directory "${HOST_PWD}" | ||||
|  | ||||
| # Download go modules | ||||
| COPY go.mod . | ||||
| COPY go.sum . | ||||
| RUN GO111MODULE=on GOPROXY=https://proxy.golang.org go mod download | ||||
|  | ||||
| COPY . /go/src/github.com/containous/traefik | ||||
| COPY . /go/src/github.com/traefik/traefik | ||||
|   | ||||
| @@ -3,8 +3,8 @@ package cmd | ||||
| import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/containous/traefik/v2/pkg/config/static" | ||||
| 	"github.com/containous/traefik/v2/pkg/types" | ||||
| 	ptypes "github.com/traefik/paerser/types" | ||||
| 	"github.com/traefik/traefik/v2/pkg/config/static" | ||||
| ) | ||||
|  | ||||
| // TraefikCmdConfiguration wraps the static configuration and extra parameters. | ||||
| @@ -23,7 +23,7 @@ func NewTraefikConfiguration() *TraefikCmdConfiguration { | ||||
| 			}, | ||||
| 			EntryPoints: make(static.EntryPoints), | ||||
| 			Providers: &static.Providers{ | ||||
| 				ProvidersThrottleDuration: types.Duration(2 * time.Second), | ||||
| 				ProvidersThrottleDuration: ptypes.Duration(2 * time.Second), | ||||
| 			}, | ||||
| 			ServersTransport: &static.ServersTransport{ | ||||
| 				MaxIdleConnsPerHost: 200, | ||||
|   | ||||
| @@ -1,22 +0,0 @@ | ||||
| package cmd | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"os" | ||||
| 	"os/signal" | ||||
| 	"syscall" | ||||
| ) | ||||
|  | ||||
| // ContextWithSignal creates a context canceled when SIGINT or SIGTERM are notified. | ||||
| func ContextWithSignal(ctx context.Context) context.Context { | ||||
| 	newCtx, cancel := context.WithCancel(ctx) | ||||
| 	signals := make(chan os.Signal) | ||||
| 	signal.Notify(signals, syscall.SIGINT, syscall.SIGTERM) | ||||
| 	go func() { | ||||
| 		select { | ||||
| 		case <-signals: | ||||
| 			cancel() | ||||
| 		} | ||||
| 	}() | ||||
| 	return newCtx | ||||
| } | ||||
| @@ -7,8 +7,8 @@ import ( | ||||
| 	"os" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/containous/traefik/v2/pkg/cli" | ||||
| 	"github.com/containous/traefik/v2/pkg/config/static" | ||||
| 	"github.com/traefik/paerser/cli" | ||||
| 	"github.com/traefik/traefik/v2/pkg/config/static" | ||||
| ) | ||||
|  | ||||
| // NewCmd builds a new HealthCheck command. | ||||
| @@ -64,7 +64,7 @@ func Do(staticConfiguration static.Configuration) (*http.Response, error) { | ||||
| 	client := &http.Client{Timeout: 5 * time.Second} | ||||
| 	protocol := "http" | ||||
|  | ||||
| 	// FIXME Handle TLS on ping etc... | ||||
| 	// TODO Handle TLS on ping etc... | ||||
| 	// if pingEntryPoint.TLS != nil { | ||||
| 	// 	protocol = "https" | ||||
| 	// 	tr := &http.Transport{ | ||||
|   | ||||
							
								
								
									
										332
									
								
								cmd/internal/gen/centrifuge.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										332
									
								
								cmd/internal/gen/centrifuge.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,332 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"go/format" | ||||
| 	"go/importer" | ||||
| 	"go/token" | ||||
| 	"go/types" | ||||
| 	"io" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"path" | ||||
| 	"path/filepath" | ||||
| 	"reflect" | ||||
| 	"slices" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
|  | ||||
| 	"golang.org/x/tools/imports" | ||||
| ) | ||||
|  | ||||
| // File a kind of AST element that represents a file. | ||||
| type File struct { | ||||
| 	Package  string | ||||
| 	Imports  []string | ||||
| 	Elements []Element | ||||
| } | ||||
|  | ||||
| // Element is a simplified version of a symbol. | ||||
| type Element struct { | ||||
| 	Name  string | ||||
| 	Value string | ||||
| } | ||||
|  | ||||
| // Centrifuge a centrifuge. | ||||
| // Generate Go Structures from Go structures. | ||||
| type Centrifuge struct { | ||||
| 	IncludedImports []string | ||||
| 	ExcludedTypes   []string | ||||
| 	ExcludedFiles   []string | ||||
|  | ||||
| 	TypeCleaner    func(types.Type, string) string | ||||
| 	PackageCleaner func(string) string | ||||
|  | ||||
| 	rootPkg string | ||||
| 	fileSet *token.FileSet | ||||
| 	pkg     *types.Package | ||||
| } | ||||
|  | ||||
| // NewCentrifuge creates a new Centrifuge. | ||||
| func NewCentrifuge(rootPkg string) (*Centrifuge, error) { | ||||
| 	fileSet := token.NewFileSet() | ||||
|  | ||||
| 	pkg, err := importer.ForCompiler(fileSet, "source", nil).Import(rootPkg) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return &Centrifuge{ | ||||
| 		fileSet: fileSet, | ||||
| 		pkg:     pkg, | ||||
| 		rootPkg: rootPkg, | ||||
|  | ||||
| 		TypeCleaner: func(typ types.Type, _ string) string { | ||||
| 			return typ.String() | ||||
| 		}, | ||||
| 		PackageCleaner: func(s string) string { | ||||
| 			return s | ||||
| 		}, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // Run runs the code extraction and the code generation. | ||||
| func (c Centrifuge) Run(dest string, pkgName string) error { | ||||
| 	files := c.run(c.pkg.Scope(), c.rootPkg, pkgName) | ||||
|  | ||||
| 	err := fileWriter{baseDir: dest}.Write(files) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	for _, p := range c.pkg.Imports() { | ||||
| 		if slices.Contains(c.IncludedImports, p.Path()) { | ||||
| 			fls := c.run(p.Scope(), p.Path(), p.Name()) | ||||
|  | ||||
| 			err = fileWriter{baseDir: filepath.Join(dest, p.Name())}.Write(fls) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (c Centrifuge) run(sc *types.Scope, rootPkg string, pkgName string) map[string]*File { | ||||
| 	files := map[string]*File{} | ||||
|  | ||||
| 	for _, name := range sc.Names() { | ||||
| 		if slices.Contains(c.ExcludedTypes, name) { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		o := sc.Lookup(name) | ||||
| 		if !o.Exported() { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		filename := filepath.Base(c.fileSet.File(o.Pos()).Name()) | ||||
| 		if slices.Contains(c.ExcludedFiles, path.Join(rootPkg, filename)) { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		fl, ok := files[filename] | ||||
| 		if !ok { | ||||
| 			files[filename] = &File{Package: pkgName} | ||||
| 			fl = files[filename] | ||||
| 		} | ||||
|  | ||||
| 		elt := Element{ | ||||
| 			Name: name, | ||||
| 		} | ||||
|  | ||||
| 		switch ob := o.(type) { | ||||
| 		case *types.TypeName: | ||||
|  | ||||
| 			switch obj := ob.Type().(*types.Named).Underlying().(type) { | ||||
| 			case *types.Struct: | ||||
| 				elt.Value = c.writeStruct(name, obj, rootPkg, fl) | ||||
|  | ||||
| 			case *types.Map: | ||||
| 				elt.Value = fmt.Sprintf("type %s map[%s]%s\n", name, obj.Key().String(), c.TypeCleaner(obj.Elem(), rootPkg)) | ||||
|  | ||||
| 			case *types.Slice: | ||||
| 				elt.Value = fmt.Sprintf("type %s []%v\n", name, c.TypeCleaner(obj.Elem(), rootPkg)) | ||||
|  | ||||
| 			case *types.Basic: | ||||
| 				elt.Value = fmt.Sprintf("type %s %v\n", name, obj.Name()) | ||||
|  | ||||
| 			default: | ||||
| 				log.Printf("OTHER TYPE::: %s %T\n", name, o.Type().(*types.Named).Underlying()) | ||||
| 				continue | ||||
| 			} | ||||
|  | ||||
| 		default: | ||||
| 			log.Printf("OTHER::: %s %T\n", name, o) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		if len(elt.Value) > 0 { | ||||
| 			fl.Elements = append(fl.Elements, elt) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return files | ||||
| } | ||||
|  | ||||
| func (c Centrifuge) writeStruct(name string, obj *types.Struct, rootPkg string, elt *File) string { | ||||
| 	b := strings.Builder{} | ||||
| 	b.WriteString(fmt.Sprintf("type %s struct {\n", name)) | ||||
|  | ||||
| 	for i := 0; i < obj.NumFields(); i++ { | ||||
| 		field := obj.Field(i) | ||||
|  | ||||
| 		if !field.Exported() { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		fPkg := c.PackageCleaner(extractPackage(field.Type())) | ||||
| 		if fPkg != "" && fPkg != rootPkg { | ||||
| 			elt.Imports = append(elt.Imports, fPkg) | ||||
| 		} | ||||
|  | ||||
| 		fType := c.TypeCleaner(field.Type(), rootPkg) | ||||
|  | ||||
| 		if field.Embedded() { | ||||
| 			b.WriteString(fmt.Sprintf("\t%s\n", fType)) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		values, ok := lookupTagValue(obj.Tag(i), "json") | ||||
| 		if len(values) > 0 && values[0] == "-" { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		b.WriteString(fmt.Sprintf("\t%s %s", field.Name(), fType)) | ||||
|  | ||||
| 		if ok { | ||||
| 			b.WriteString(fmt.Sprintf(" `json:\"%s\"`", strings.Join(values, ","))) | ||||
| 		} | ||||
|  | ||||
| 		b.WriteString("\n") | ||||
| 	} | ||||
|  | ||||
| 	b.WriteString("}\n") | ||||
|  | ||||
| 	return b.String() | ||||
| } | ||||
|  | ||||
| func lookupTagValue(raw, key string) ([]string, bool) { | ||||
| 	value, ok := reflect.StructTag(raw).Lookup(key) | ||||
| 	if !ok { | ||||
| 		return nil, ok | ||||
| 	} | ||||
|  | ||||
| 	values := strings.Split(value, ",") | ||||
|  | ||||
| 	if len(values) < 1 { | ||||
| 		return nil, true | ||||
| 	} | ||||
|  | ||||
| 	return values, true | ||||
| } | ||||
|  | ||||
| func extractPackage(t types.Type) string { | ||||
| 	switch tu := t.(type) { | ||||
| 	case *types.Named: | ||||
| 		return tu.Obj().Pkg().Path() | ||||
|  | ||||
| 	case *types.Slice: | ||||
| 		if v, ok := tu.Elem().(*types.Named); ok { | ||||
| 			return v.Obj().Pkg().Path() | ||||
| 		} | ||||
| 		return "" | ||||
|  | ||||
| 	case *types.Map: | ||||
| 		if v, ok := tu.Elem().(*types.Named); ok { | ||||
| 			return v.Obj().Pkg().Path() | ||||
| 		} | ||||
| 		return "" | ||||
|  | ||||
| 	case *types.Pointer: | ||||
| 		return extractPackage(tu.Elem()) | ||||
|  | ||||
| 	default: | ||||
| 		return "" | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type fileWriter struct { | ||||
| 	baseDir string | ||||
| } | ||||
|  | ||||
| func (f fileWriter) Write(files map[string]*File) error { | ||||
| 	err := os.MkdirAll(f.baseDir, 0o755) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	for name, file := range files { | ||||
| 		err = f.writeFile(name, file) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (f fileWriter) writeFile(name string, desc *File) error { | ||||
| 	if len(desc.Elements) == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	filename := filepath.Join(f.baseDir, name) | ||||
|  | ||||
| 	file, err := os.Create(filename) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("failed to create file: %w", err) | ||||
| 	} | ||||
|  | ||||
| 	defer func() { _ = file.Close() }() | ||||
|  | ||||
| 	b := bytes.NewBufferString("package ") | ||||
| 	b.WriteString(desc.Package) | ||||
| 	b.WriteString("\n") | ||||
| 	b.WriteString("// Code generated by centrifuge. DO NOT EDIT.\n") | ||||
|  | ||||
| 	b.WriteString("\n") | ||||
| 	f.writeImports(b, desc.Imports) | ||||
| 	b.WriteString("\n") | ||||
|  | ||||
| 	for _, elt := range desc.Elements { | ||||
| 		b.WriteString(elt.Value) | ||||
| 		b.WriteString("\n") | ||||
| 	} | ||||
|  | ||||
| 	// gofmt | ||||
| 	source, err := format.Source(b.Bytes()) | ||||
| 	if err != nil { | ||||
| 		log.Println(b.String()) | ||||
| 		return fmt.Errorf("failed to format sources: %w", err) | ||||
| 	} | ||||
|  | ||||
| 	// goimports | ||||
| 	process, err := imports.Process(filename, source, nil) | ||||
| 	if err != nil { | ||||
| 		log.Println(string(source)) | ||||
| 		return fmt.Errorf("failed to format imports: %w", err) | ||||
| 	} | ||||
|  | ||||
| 	_, err = file.Write(process) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (f fileWriter) writeImports(b io.StringWriter, imports []string) { | ||||
| 	if len(imports) == 0 { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	uniq := map[string]struct{}{} | ||||
|  | ||||
| 	sort.Strings(imports) | ||||
|  | ||||
| 	_, _ = b.WriteString("import (\n") | ||||
| 	for _, s := range imports { | ||||
| 		if _, exist := uniq[s]; exist { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		uniq[s] = struct{}{} | ||||
|  | ||||
| 		_, _ = b.WriteString(fmt.Sprintf(`	"%s"`+"\n", s)) | ||||
| 	} | ||||
|  | ||||
| 	_, _ = b.WriteString(")\n") | ||||
| } | ||||
							
								
								
									
										124
									
								
								cmd/internal/gen/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								cmd/internal/gen/main.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"go/build" | ||||
| 	"go/types" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"path" | ||||
| 	"path/filepath" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| const rootPkg = "github.com/traefik/traefik/v2/pkg/config/dynamic" | ||||
|  | ||||
| const ( | ||||
| 	destModuleName = "github.com/traefik/genconf" | ||||
| 	destPkg        = "dynamic" | ||||
| ) | ||||
|  | ||||
| const marsh = `package %s | ||||
|  | ||||
| import "encoding/json" | ||||
|  | ||||
| type JSONPayload struct { | ||||
| 	*Configuration | ||||
| } | ||||
|  | ||||
| func (c JSONPayload) MarshalJSON() ([]byte, error) { | ||||
| 	if c.Configuration == nil { | ||||
| 		return nil, nil | ||||
| 	} | ||||
|  | ||||
| 	return json.Marshal(c.Configuration) | ||||
| } | ||||
| ` | ||||
|  | ||||
| // main generate Go Structures from Go structures. | ||||
| // Allows to create an external module (destModuleName) used by the plugin's providers | ||||
| // that contains Go structs of the dynamic configuration and nothing else. | ||||
| // These Go structs do not have any non-exported fields and do not rely on any external dependencies. | ||||
| func main() { | ||||
| 	dest := filepath.Join(path.Join(build.Default.GOPATH, "src"), destModuleName, destPkg) | ||||
|  | ||||
| 	log.Println("Output:", dest) | ||||
|  | ||||
| 	err := run(dest) | ||||
| 	if err != nil { | ||||
| 		log.Fatal(err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func run(dest string) error { | ||||
| 	centrifuge, err := NewCentrifuge(rootPkg) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	centrifuge.IncludedImports = []string{ | ||||
| 		"github.com/traefik/traefik/v2/pkg/tls", | ||||
| 		"github.com/traefik/traefik/v2/pkg/types", | ||||
| 	} | ||||
|  | ||||
| 	centrifuge.ExcludedTypes = []string{ | ||||
| 		// tls | ||||
| 		"CertificateStore", "Manager", | ||||
| 		// dynamic | ||||
| 		"Message", "Configurations", | ||||
| 		// types | ||||
| 		"HTTPCodeRanges", "HostResolverConfig", | ||||
| 	} | ||||
|  | ||||
| 	centrifuge.ExcludedFiles = []string{ | ||||
| 		"github.com/traefik/traefik/v2/pkg/types/logs.go", | ||||
| 		"github.com/traefik/traefik/v2/pkg/types/metrics.go", | ||||
| 	} | ||||
|  | ||||
| 	centrifuge.TypeCleaner = cleanType | ||||
| 	centrifuge.PackageCleaner = cleanPackage | ||||
|  | ||||
| 	err = centrifuge.Run(dest, destPkg) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return os.WriteFile(filepath.Join(dest, "marshaler.go"), []byte(fmt.Sprintf(marsh, destPkg)), 0o666) | ||||
| } | ||||
|  | ||||
| func cleanType(typ types.Type, base string) string { | ||||
| 	if typ.String() == "github.com/traefik/traefik/v2/pkg/tls.FileOrContent" { | ||||
| 		return "string" | ||||
| 	} | ||||
|  | ||||
| 	if typ.String() == "[]github.com/traefik/traefik/v2/pkg/tls.FileOrContent" { | ||||
| 		return "[]string" | ||||
| 	} | ||||
|  | ||||
| 	if typ.String() == "github.com/traefik/paerser/types.Duration" { | ||||
| 		return "string" | ||||
| 	} | ||||
|  | ||||
| 	if strings.Contains(typ.String(), base) { | ||||
| 		return strings.ReplaceAll(typ.String(), base+".", "") | ||||
| 	} | ||||
|  | ||||
| 	if strings.Contains(typ.String(), "github.com/traefik/traefik/v2/pkg/") { | ||||
| 		return strings.ReplaceAll(typ.String(), "github.com/traefik/traefik/v2/pkg/", "") | ||||
| 	} | ||||
|  | ||||
| 	return typ.String() | ||||
| } | ||||
|  | ||||
| func cleanPackage(src string) string { | ||||
| 	switch src { | ||||
| 	case "github.com/traefik/paerser/types": | ||||
| 		return "" | ||||
| 	case "github.com/traefik/traefik/v2/pkg/tls": | ||||
| 		return path.Join(destModuleName, destPkg, "tls") | ||||
| 	case "github.com/traefik/traefik/v2/pkg/types": | ||||
| 		return path.Join(destModuleName, destPkg, "types") | ||||
| 	default: | ||||
| 		return src | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										83
									
								
								cmd/traefik/plugins.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								cmd/traefik/plugins.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,83 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"github.com/traefik/traefik/v2/pkg/config/static" | ||||
| 	"github.com/traefik/traefik/v2/pkg/plugins" | ||||
| ) | ||||
|  | ||||
| const outputDir = "./plugins-storage/" | ||||
|  | ||||
| func createPluginBuilder(staticConfiguration *static.Configuration) (*plugins.Builder, error) { | ||||
| 	client, plgs, localPlgs, err := initPlugins(staticConfiguration) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return plugins.NewBuilder(client, plgs, localPlgs) | ||||
| } | ||||
|  | ||||
| func initPlugins(staticCfg *static.Configuration) (*plugins.Client, map[string]plugins.Descriptor, map[string]plugins.LocalDescriptor, error) { | ||||
| 	err := checkUniquePluginNames(staticCfg.Experimental) | ||||
| 	if err != nil { | ||||
| 		return nil, nil, nil, err | ||||
| 	} | ||||
|  | ||||
| 	var client *plugins.Client | ||||
| 	plgs := map[string]plugins.Descriptor{} | ||||
|  | ||||
| 	if hasPlugins(staticCfg) { | ||||
| 		opts := plugins.ClientOptions{ | ||||
| 			Output: outputDir, | ||||
| 		} | ||||
|  | ||||
| 		var err error | ||||
| 		client, err = plugins.NewClient(opts) | ||||
| 		if err != nil { | ||||
| 			return nil, nil, nil, fmt.Errorf("unable to create plugins client: %w", err) | ||||
| 		} | ||||
|  | ||||
| 		err = plugins.SetupRemotePlugins(client, staticCfg.Experimental.Plugins) | ||||
| 		if err != nil { | ||||
| 			return nil, nil, nil, fmt.Errorf("unable to set up plugins environment: %w", err) | ||||
| 		} | ||||
|  | ||||
| 		plgs = staticCfg.Experimental.Plugins | ||||
| 	} | ||||
|  | ||||
| 	localPlgs := map[string]plugins.LocalDescriptor{} | ||||
|  | ||||
| 	if hasLocalPlugins(staticCfg) { | ||||
| 		err := plugins.SetupLocalPlugins(staticCfg.Experimental.LocalPlugins) | ||||
| 		if err != nil { | ||||
| 			return nil, nil, nil, err | ||||
| 		} | ||||
|  | ||||
| 		localPlgs = staticCfg.Experimental.LocalPlugins | ||||
| 	} | ||||
|  | ||||
| 	return client, plgs, localPlgs, nil | ||||
| } | ||||
|  | ||||
| func checkUniquePluginNames(e *static.Experimental) error { | ||||
| 	if e == nil { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	for s := range e.LocalPlugins { | ||||
| 		if _, ok := e.Plugins[s]; ok { | ||||
| 			return fmt.Errorf("the plugin's name %q must be unique", s) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func hasPlugins(staticCfg *static.Configuration) bool { | ||||
| 	return staticCfg.Experimental != nil && len(staticCfg.Experimental.Plugins) > 0 | ||||
| } | ||||
|  | ||||
| func hasLocalPlugins(staticCfg *static.Configuration) bool { | ||||
| 	return staticCfg.Experimental != nil && len(staticCfg.Experimental.LocalPlugins) > 0 | ||||
| } | ||||
| @@ -2,48 +2,55 @@ package main | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"crypto/x509" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	stdlog "log" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"os/signal" | ||||
| 	"path/filepath" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| 	"syscall" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/containous/traefik/v2/autogen/genstatic" | ||||
| 	"github.com/containous/traefik/v2/cmd" | ||||
| 	"github.com/containous/traefik/v2/cmd/healthcheck" | ||||
| 	cmdVersion "github.com/containous/traefik/v2/cmd/version" | ||||
| 	"github.com/containous/traefik/v2/pkg/cli" | ||||
| 	"github.com/containous/traefik/v2/pkg/collector" | ||||
| 	"github.com/containous/traefik/v2/pkg/config/dynamic" | ||||
| 	"github.com/containous/traefik/v2/pkg/config/static" | ||||
| 	"github.com/containous/traefik/v2/pkg/log" | ||||
| 	"github.com/containous/traefik/v2/pkg/metrics" | ||||
| 	"github.com/containous/traefik/v2/pkg/middlewares/accesslog" | ||||
| 	"github.com/containous/traefik/v2/pkg/provider/acme" | ||||
| 	"github.com/containous/traefik/v2/pkg/provider/aggregator" | ||||
| 	"github.com/containous/traefik/v2/pkg/provider/traefik" | ||||
| 	"github.com/containous/traefik/v2/pkg/rules" | ||||
| 	"github.com/containous/traefik/v2/pkg/safe" | ||||
| 	"github.com/containous/traefik/v2/pkg/server" | ||||
| 	"github.com/containous/traefik/v2/pkg/server/middleware" | ||||
| 	"github.com/containous/traefik/v2/pkg/server/service" | ||||
| 	traefiktls "github.com/containous/traefik/v2/pkg/tls" | ||||
| 	"github.com/containous/traefik/v2/pkg/types" | ||||
| 	"github.com/containous/traefik/v2/pkg/version" | ||||
| 	"github.com/coreos/go-systemd/daemon" | ||||
| 	assetfs "github.com/elazarl/go-bindata-assetfs" | ||||
| 	"github.com/go-acme/lego/v4/challenge" | ||||
| 	gokitmetrics "github.com/go-kit/kit/metrics" | ||||
| 	"github.com/sirupsen/logrus" | ||||
| 	"github.com/vulcand/oxy/roundrobin" | ||||
| 	"github.com/traefik/paerser/cli" | ||||
| 	"github.com/traefik/traefik/v2/cmd" | ||||
| 	"github.com/traefik/traefik/v2/cmd/healthcheck" | ||||
| 	cmdVersion "github.com/traefik/traefik/v2/cmd/version" | ||||
| 	tcli "github.com/traefik/traefik/v2/pkg/cli" | ||||
| 	"github.com/traefik/traefik/v2/pkg/collector" | ||||
| 	"github.com/traefik/traefik/v2/pkg/config/dynamic" | ||||
| 	"github.com/traefik/traefik/v2/pkg/config/runtime" | ||||
| 	"github.com/traefik/traefik/v2/pkg/config/static" | ||||
| 	"github.com/traefik/traefik/v2/pkg/log" | ||||
| 	"github.com/traefik/traefik/v2/pkg/metrics" | ||||
| 	"github.com/traefik/traefik/v2/pkg/middlewares/accesslog" | ||||
| 	"github.com/traefik/traefik/v2/pkg/provider/acme" | ||||
| 	"github.com/traefik/traefik/v2/pkg/provider/aggregator" | ||||
| 	"github.com/traefik/traefik/v2/pkg/provider/traefik" | ||||
| 	"github.com/traefik/traefik/v2/pkg/safe" | ||||
| 	"github.com/traefik/traefik/v2/pkg/server" | ||||
| 	"github.com/traefik/traefik/v2/pkg/server/middleware" | ||||
| 	"github.com/traefik/traefik/v2/pkg/server/service" | ||||
| 	traefiktls "github.com/traefik/traefik/v2/pkg/tls" | ||||
| 	"github.com/traefik/traefik/v2/pkg/tracing" | ||||
| 	"github.com/traefik/traefik/v2/pkg/tracing/jaeger" | ||||
| 	"github.com/traefik/traefik/v2/pkg/types" | ||||
| 	"github.com/traefik/traefik/v2/pkg/version" | ||||
| 	"github.com/vulcand/oxy/v2/roundrobin" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
| 	// traefik config inits | ||||
| 	tConfig := cmd.NewTraefikConfiguration() | ||||
|  | ||||
| 	loaders := []cli.ResourceLoader{&cli.FileLoader{}, &cli.FlagLoader{}, &cli.EnvLoader{}} | ||||
| 	loaders := []cli.ResourceLoader{&tcli.FileLoader{}, &tcli.FlagLoader{}, &tcli.EnvLoader{}} | ||||
|  | ||||
| 	cmdTraefik := &cli.Command{ | ||||
| 		Name: "traefik", | ||||
| @@ -101,10 +108,6 @@ func runCmd(staticConfiguration *static.Configuration) error { | ||||
| 		log.WithoutContext().Debugf("Static configuration loaded %s", string(jsonConf)) | ||||
| 	} | ||||
|  | ||||
| 	if staticConfiguration.API != nil && staticConfiguration.API.Dashboard { | ||||
| 		staticConfiguration.API.DashboardAssets = &assetfs.AssetFS{Asset: genstatic.Asset, AssetInfo: genstatic.AssetInfo, AssetDir: genstatic.AssetDir, Prefix: "static"} | ||||
| 	} | ||||
|  | ||||
| 	if staticConfiguration.Global.CheckNewVersion { | ||||
| 		checkNewVersion() | ||||
| 	} | ||||
| @@ -116,7 +119,7 @@ func runCmd(staticConfiguration *static.Configuration) error { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	ctx := cmd.ContextWithSignal(context.Background()) | ||||
| 	ctx, _ := signal.NotifyContext(context.Background(), syscall.SIGINT, syscall.SIGTERM) | ||||
|  | ||||
| 	if staticConfiguration.Ping != nil { | ||||
| 		staticConfiguration.Ping.WithContext(ctx) | ||||
| @@ -162,21 +165,33 @@ func runCmd(staticConfiguration *static.Configuration) error { | ||||
| } | ||||
|  | ||||
| func setupServer(staticConfiguration *static.Configuration) (*server.Server, error) { | ||||
| 	rules.EnableDomainFronting(staticConfiguration.Global.InsecureSNI) | ||||
|  | ||||
| 	providerAggregator := aggregator.NewProviderAggregator(*staticConfiguration.Providers) | ||||
|  | ||||
| 	ctx := context.Background() | ||||
| 	routinesPool := safe.NewPool(ctx) | ||||
|  | ||||
| 	// adds internal provider | ||||
| 	err := providerAggregator.AddProvider(traefik.New(*staticConfiguration)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// ACME | ||||
|  | ||||
| 	tlsManager := traefiktls.NewManager() | ||||
| 	httpChallengeProvider := acme.NewChallengeHTTP() | ||||
|  | ||||
| 	acmeProviders := initACMEProvider(staticConfiguration, &providerAggregator, tlsManager) | ||||
| 	tlsChallengeProvider := acme.NewChallengeTLSALPN() | ||||
| 	err = providerAggregator.AddProvider(tlsChallengeProvider) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	serverEntryPointsTCP, err := server.NewTCPEntryPoints(staticConfiguration.EntryPoints) | ||||
| 	acmeProviders := initACMEProvider(staticConfiguration, &providerAggregator, tlsManager, httpChallengeProvider, tlsChallengeProvider) | ||||
|  | ||||
| 	// Entrypoints | ||||
|  | ||||
| 	serverEntryPointsTCP, err := server.NewTCPEntryPoints(staticConfiguration.EntryPoints, staticConfiguration.HostResolver) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @@ -186,66 +201,114 @@ func setupServer(staticConfiguration *static.Configuration) (*server.Server, err | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	ctx := context.Background() | ||||
| 	routinesPool := safe.NewPool(ctx) | ||||
| 	if staticConfiguration.Pilot != nil { | ||||
| 		log.WithoutContext().Warn("Traefik Pilot has been removed.") | ||||
| 	} | ||||
|  | ||||
| 	metricsRegistry := registerMetricClients(staticConfiguration.Metrics) | ||||
| 	accessLog := setupAccessLog(staticConfiguration.AccessLog) | ||||
| 	chainBuilder := middleware.NewChainBuilder(*staticConfiguration, metricsRegistry, accessLog) | ||||
| 	managerFactory := service.NewManagerFactory(*staticConfiguration, routinesPool, metricsRegistry) | ||||
| 	routerFactory := server.NewRouterFactory(*staticConfiguration, managerFactory, tlsManager, chainBuilder) | ||||
| 	if staticConfiguration.API != nil { | ||||
| 		version.DisableDashboardAd = staticConfiguration.API.DisableDashboardAd | ||||
| 	} | ||||
|  | ||||
| 	var defaultEntryPoints []string | ||||
| 	for name, cfg := range staticConfiguration.EntryPoints { | ||||
| 		protocol, err := cfg.GetProtocol() | ||||
| 		if err != nil { | ||||
| 			// Should never happen because Traefik should not start if protocol is invalid. | ||||
| 			log.WithoutContext().Errorf("Invalid protocol: %v", err) | ||||
| 	// Plugins | ||||
|  | ||||
| 	pluginBuilder, err := createPluginBuilder(staticConfiguration) | ||||
| 	if err != nil { | ||||
| 		log.WithoutContext().WithError(err).Error("Plugins are disabled because an error has occurred.") | ||||
| 	} | ||||
|  | ||||
| 	// Providers plugins | ||||
|  | ||||
| 	for name, conf := range staticConfiguration.Providers.Plugin { | ||||
| 		if pluginBuilder == nil { | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		if protocol != "udp" && name != static.DefaultInternalEntryPointName { | ||||
| 			defaultEntryPoints = append(defaultEntryPoints, name) | ||||
| 		p, err := pluginBuilder.BuildProvider(name, conf) | ||||
| 		if err != nil { | ||||
| 			return nil, fmt.Errorf("plugin: failed to build provider: %w", err) | ||||
| 		} | ||||
|  | ||||
| 		err = providerAggregator.AddProvider(p) | ||||
| 		if err != nil { | ||||
| 			return nil, fmt.Errorf("plugin: failed to add provider: %w", err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	sort.Strings(defaultEntryPoints) | ||||
| 	// Metrics | ||||
|  | ||||
| 	metricRegistries := registerMetricClients(staticConfiguration.Metrics) | ||||
| 	metricsRegistry := metrics.NewMultiRegistry(metricRegistries) | ||||
|  | ||||
| 	// Service manager factory | ||||
|  | ||||
| 	roundTripperManager := service.NewRoundTripperManager() | ||||
| 	acmeHTTPHandler := getHTTPChallengeHandler(acmeProviders, httpChallengeProvider) | ||||
| 	managerFactory := service.NewManagerFactory(*staticConfiguration, routinesPool, metricsRegistry, roundTripperManager, acmeHTTPHandler) | ||||
|  | ||||
| 	// Router factory | ||||
|  | ||||
| 	accessLog := setupAccessLog(staticConfiguration.AccessLog) | ||||
| 	tracer := setupTracing(staticConfiguration.Tracing) | ||||
|  | ||||
| 	chainBuilder := middleware.NewChainBuilder(metricsRegistry, accessLog, tracer) | ||||
| 	routerFactory := server.NewRouterFactory(*staticConfiguration, managerFactory, tlsManager, chainBuilder, pluginBuilder, metricsRegistry) | ||||
|  | ||||
| 	// Watcher | ||||
|  | ||||
| 	watcher := server.NewConfigurationWatcher( | ||||
| 		routinesPool, | ||||
| 		providerAggregator, | ||||
| 		time.Duration(staticConfiguration.Providers.ProvidersThrottleDuration), | ||||
| 		defaultEntryPoints, | ||||
| 		getDefaultsEntrypoints(staticConfiguration), | ||||
| 		"internal", | ||||
| 	) | ||||
|  | ||||
| 	// TLS | ||||
| 	watcher.AddListener(func(conf dynamic.Configuration) { | ||||
| 		ctx := context.Background() | ||||
| 		tlsManager.UpdateConfigs(ctx, conf.TLS.Stores, conf.TLS.Options, conf.TLS.Certificates) | ||||
|  | ||||
| 		gauge := metricsRegistry.TLSCertsNotAfterTimestampGauge() | ||||
| 		for _, certificate := range tlsManager.GetServerCertificates() { | ||||
| 			appendCertMetric(gauge, certificate) | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	// Metrics | ||||
| 	watcher.AddListener(func(_ dynamic.Configuration) { | ||||
| 		metricsRegistry.ConfigReloadsCounter().Add(1) | ||||
| 		metricsRegistry.LastConfigReloadSuccessGauge().Set(float64(time.Now().Unix())) | ||||
| 	}) | ||||
|  | ||||
| 	watcher.AddListener(switchRouter(routerFactory, acmeProviders, serverEntryPointsTCP, serverEntryPointsUDP)) | ||||
|  | ||||
| 	// Server Transports | ||||
| 	watcher.AddListener(func(conf dynamic.Configuration) { | ||||
| 		if metricsRegistry.IsEpEnabled() || metricsRegistry.IsSvcEnabled() { | ||||
| 			var eps []string | ||||
| 			for key := range serverEntryPointsTCP { | ||||
| 				eps = append(eps, key) | ||||
| 			} | ||||
|  | ||||
| 			metrics.OnConfigurationUpdate(conf, eps) | ||||
| 		} | ||||
| 		roundTripperManager.Update(conf.HTTP.ServersTransports) | ||||
| 	}) | ||||
|  | ||||
| 	// Switch router | ||||
| 	watcher.AddListener(switchRouter(routerFactory, serverEntryPointsTCP, serverEntryPointsUDP)) | ||||
|  | ||||
| 	// Metrics | ||||
| 	if metricsRegistry.IsEpEnabled() || metricsRegistry.IsRouterEnabled() || metricsRegistry.IsSvcEnabled() { | ||||
| 		var eps []string | ||||
| 		for key := range serverEntryPointsTCP { | ||||
| 			eps = append(eps, key) | ||||
| 		} | ||||
| 		watcher.AddListener(func(conf dynamic.Configuration) { | ||||
| 			metrics.OnConfigurationUpdate(conf, eps) | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	// TLS challenge | ||||
| 	watcher.AddListener(tlsChallengeProvider.ListenConfiguration) | ||||
|  | ||||
| 	// ACME | ||||
| 	resolverNames := map[string]struct{}{} | ||||
| 	for _, p := range acmeProviders { | ||||
| 		resolverNames[p.ResolverName] = struct{}{} | ||||
| 		watcher.AddListener(p.ListenConfiguration) | ||||
| 	} | ||||
|  | ||||
| 	// Certificate resolver logs | ||||
| 	watcher.AddListener(func(config dynamic.Configuration) { | ||||
| 		for rtName, rt := range config.HTTP.Routers { | ||||
| 			if rt.TLS == nil || rt.TLS.CertResolver == "" { | ||||
| @@ -261,63 +324,86 @@ func setupServer(staticConfiguration *static.Configuration) (*server.Server, err | ||||
| 	return server.NewServer(routinesPool, serverEntryPointsTCP, serverEntryPointsUDP, watcher, chainBuilder, accessLog), nil | ||||
| } | ||||
|  | ||||
| func switchRouter(routerFactory *server.RouterFactory, acmeProviders []*acme.Provider, serverEntryPointsTCP server.TCPEntryPoints, serverEntryPointsUDP server.UDPEntryPoints) func(conf dynamic.Configuration) { | ||||
| 	return func(conf dynamic.Configuration) { | ||||
| 		routers, udpRouters := routerFactory.CreateRouters(conf) | ||||
| 		for entryPointName, rt := range routers { | ||||
| 			for _, p := range acmeProviders { | ||||
| 				if p != nil && p.HTTPChallenge != nil && p.HTTPChallenge.EntryPoint == entryPointName { | ||||
| 					rt.HTTPHandler(p.CreateHandler(rt.GetHTTPHandler())) | ||||
| 					break | ||||
| 				} | ||||
| 			} | ||||
| func getHTTPChallengeHandler(acmeProviders []*acme.Provider, httpChallengeProvider http.Handler) http.Handler { | ||||
| 	var acmeHTTPHandler http.Handler | ||||
| 	for _, p := range acmeProviders { | ||||
| 		if p != nil && p.HTTPChallenge != nil { | ||||
| 			acmeHTTPHandler = httpChallengeProvider | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	return acmeHTTPHandler | ||||
| } | ||||
|  | ||||
| func getDefaultsEntrypoints(staticConfiguration *static.Configuration) []string { | ||||
| 	var defaultEntryPoints []string | ||||
| 	for name, cfg := range staticConfiguration.EntryPoints { | ||||
| 		protocol, err := cfg.GetProtocol() | ||||
| 		if err != nil { | ||||
| 			// Should never happen because Traefik should not start if protocol is invalid. | ||||
| 			log.WithoutContext().Errorf("Invalid protocol: %v", err) | ||||
| 		} | ||||
|  | ||||
| 		if protocol != "udp" && name != static.DefaultInternalEntryPointName { | ||||
| 			defaultEntryPoints = append(defaultEntryPoints, name) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	sort.Strings(defaultEntryPoints) | ||||
| 	return defaultEntryPoints | ||||
| } | ||||
|  | ||||
| func switchRouter(routerFactory *server.RouterFactory, serverEntryPointsTCP server.TCPEntryPoints, serverEntryPointsUDP server.UDPEntryPoints) func(conf dynamic.Configuration) { | ||||
| 	return func(conf dynamic.Configuration) { | ||||
| 		rtConf := runtime.NewConfig(conf) | ||||
|  | ||||
| 		routers, udpRouters := routerFactory.CreateRouters(rtConf) | ||||
|  | ||||
| 		serverEntryPointsTCP.Switch(routers) | ||||
| 		serverEntryPointsUDP.Switch(udpRouters) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // initACMEProvider creates an acme provider from the ACME part of globalConfiguration. | ||||
| func initACMEProvider(c *static.Configuration, providerAggregator *aggregator.ProviderAggregator, tlsManager *traefiktls.Manager) []*acme.Provider { | ||||
| 	challengeStore := acme.NewLocalChallengeStore() | ||||
| func initACMEProvider(c *static.Configuration, providerAggregator *aggregator.ProviderAggregator, tlsManager *traefiktls.Manager, httpChallengeProvider, tlsChallengeProvider challenge.Provider) []*acme.Provider { | ||||
| 	localStores := map[string]*acme.LocalStore{} | ||||
|  | ||||
| 	var resolvers []*acme.Provider | ||||
| 	for name, resolver := range c.CertificatesResolvers { | ||||
| 		if resolver.ACME != nil { | ||||
| 			if localStores[resolver.ACME.Storage] == nil { | ||||
| 				localStores[resolver.ACME.Storage] = acme.NewLocalStore(resolver.ACME.Storage) | ||||
| 			} | ||||
|  | ||||
| 			p := &acme.Provider{ | ||||
| 				Configuration:  resolver.ACME, | ||||
| 				Store:          localStores[resolver.ACME.Storage], | ||||
| 				ChallengeStore: challengeStore, | ||||
| 				ResolverName:   name, | ||||
| 			} | ||||
|  | ||||
| 			if err := providerAggregator.AddProvider(p); err != nil { | ||||
| 				log.WithoutContext().Errorf("The ACME resolver %q is skipped from the resolvers list because: %v", name, err) | ||||
| 				continue | ||||
| 			} | ||||
|  | ||||
| 			p.SetTLSManager(tlsManager) | ||||
|  | ||||
| 			if p.TLSChallenge != nil { | ||||
| 				tlsManager.TLSAlpnGetter = p.GetTLSALPNCertificate | ||||
| 			} | ||||
|  | ||||
| 			p.SetConfigListenerChan(make(chan dynamic.Configuration)) | ||||
|  | ||||
| 			resolvers = append(resolvers, p) | ||||
| 		if resolver.ACME == nil { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		if localStores[resolver.ACME.Storage] == nil { | ||||
| 			localStores[resolver.ACME.Storage] = acme.NewLocalStore(resolver.ACME.Storage) | ||||
| 		} | ||||
|  | ||||
| 		p := &acme.Provider{ | ||||
| 			Configuration:         resolver.ACME, | ||||
| 			Store:                 localStores[resolver.ACME.Storage], | ||||
| 			ResolverName:          name, | ||||
| 			HTTPChallengeProvider: httpChallengeProvider, | ||||
| 			TLSChallengeProvider:  tlsChallengeProvider, | ||||
| 		} | ||||
|  | ||||
| 		if err := providerAggregator.AddProvider(p); err != nil { | ||||
| 			log.WithoutContext().Errorf("The ACME resolver %q is skipped from the resolvers list because: %v", name, err) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		p.SetTLSManager(tlsManager) | ||||
|  | ||||
| 		p.SetConfigListenerChan(make(chan dynamic.Configuration)) | ||||
|  | ||||
| 		resolvers = append(resolvers, p) | ||||
| 	} | ||||
|  | ||||
| 	return resolvers | ||||
| } | ||||
|  | ||||
| func registerMetricClients(metricsConfig *types.Metrics) metrics.Registry { | ||||
| func registerMetricClients(metricsConfig *types.Metrics) []metrics.Registry { | ||||
| 	if metricsConfig == nil { | ||||
| 		return metrics.NewVoidRegistry() | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	var registries []metrics.Registry | ||||
| @@ -352,7 +438,31 @@ func registerMetricClients(metricsConfig *types.Metrics) metrics.Registry { | ||||
| 			metricsConfig.InfluxDB.Address, metricsConfig.InfluxDB.PushInterval) | ||||
| 	} | ||||
|  | ||||
| 	return metrics.NewMultiRegistry(registries) | ||||
| 	if metricsConfig.InfluxDB2 != nil { | ||||
| 		ctx := log.With(context.Background(), log.Str(log.MetricsProviderName, "influxdb2")) | ||||
| 		influxDB2Register := metrics.RegisterInfluxDB2(ctx, metricsConfig.InfluxDB2) | ||||
| 		if influxDB2Register != nil { | ||||
| 			registries = append(registries, influxDB2Register) | ||||
| 			log.FromContext(ctx).Debugf("Configured InfluxDB v2 metrics: pushing to %s (%s org/%s bucket) once every %s", | ||||
| 				metricsConfig.InfluxDB2.Address, metricsConfig.InfluxDB2.Org, metricsConfig.InfluxDB2.Bucket, metricsConfig.InfluxDB2.PushInterval) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return registries | ||||
| } | ||||
|  | ||||
| func appendCertMetric(gauge gokitmetrics.Gauge, certificate *x509.Certificate) { | ||||
| 	sort.Strings(certificate.DNSNames) | ||||
|  | ||||
| 	labels := []string{ | ||||
| 		"cn", certificate.Subject.CommonName, | ||||
| 		"serial", certificate.SerialNumber.String(), | ||||
| 		"sans", strings.Join(certificate.DNSNames, ","), | ||||
| 	} | ||||
|  | ||||
| 	notAfter := float64(certificate.NotAfter.Unix()) | ||||
|  | ||||
| 	gauge.With(labels...).Set(notAfter) | ||||
| } | ||||
|  | ||||
| func setupAccessLog(conf *types.AccessLog) *accesslog.Handler { | ||||
| @@ -362,13 +472,79 @@ func setupAccessLog(conf *types.AccessLog) *accesslog.Handler { | ||||
|  | ||||
| 	accessLoggerMiddleware, err := accesslog.NewHandler(conf) | ||||
| 	if err != nil { | ||||
| 		log.WithoutContext().Warnf("Unable to create access logger : %v", err) | ||||
| 		log.WithoutContext().Warnf("Unable to create access logger: %v", err) | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	return accessLoggerMiddleware | ||||
| } | ||||
|  | ||||
| func setupTracing(conf *static.Tracing) *tracing.Tracing { | ||||
| 	if conf == nil { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	var backend tracing.Backend | ||||
|  | ||||
| 	if conf.Jaeger != nil { | ||||
| 		backend = conf.Jaeger | ||||
| 	} | ||||
|  | ||||
| 	if conf.Zipkin != nil { | ||||
| 		if backend != nil { | ||||
| 			log.WithoutContext().Error("Multiple tracing backend are not supported: cannot create Zipkin backend.") | ||||
| 		} else { | ||||
| 			backend = conf.Zipkin | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if conf.Datadog != nil { | ||||
| 		if backend != nil { | ||||
| 			log.WithoutContext().Error("Multiple tracing backend are not supported: cannot create Datadog backend.") | ||||
| 		} else { | ||||
| 			backend = conf.Datadog | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if conf.Instana != nil { | ||||
| 		if backend != nil { | ||||
| 			log.WithoutContext().Error("Multiple tracing backend are not supported: cannot create Instana backend.") | ||||
| 		} else { | ||||
| 			backend = conf.Instana | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if conf.Haystack != nil { | ||||
| 		if backend != nil { | ||||
| 			log.WithoutContext().Error("Multiple tracing backend are not supported: cannot create Haystack backend.") | ||||
| 		} else { | ||||
| 			backend = conf.Haystack | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if conf.Elastic != nil { | ||||
| 		if backend != nil { | ||||
| 			log.WithoutContext().Error("Multiple tracing backend are not supported: cannot create Elastic backend.") | ||||
| 		} else { | ||||
| 			backend = conf.Elastic | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if backend == nil { | ||||
| 		log.WithoutContext().Debug("Could not initialize tracing, using Jaeger by default") | ||||
| 		defaultBackend := &jaeger.Config{} | ||||
| 		defaultBackend.SetDefaults() | ||||
| 		backend = defaultBackend | ||||
| 	} | ||||
|  | ||||
| 	tracer, err := tracing.NewTracing(conf.ServiceName, conf.SpanNameLimit, backend) | ||||
| 	if err != nil { | ||||
| 		log.WithoutContext().Warnf("Unable to create tracer: %v", err) | ||||
| 		return nil | ||||
| 	} | ||||
| 	return tracer | ||||
| } | ||||
|  | ||||
| func configureLogging(staticConfiguration *static.Configuration) { | ||||
| 	// configure default log flags | ||||
| 	stdlog.SetFlags(stdlog.Lshortfile | stdlog.LstdFlags) | ||||
| @@ -438,13 +614,13 @@ func stats(staticConfiguration *static.Configuration) { | ||||
| 		logger.Info(`Stats collection is enabled.`) | ||||
| 		logger.Info(`Many thanks for contributing to Traefik's improvement by allowing us to receive anonymous information from your configuration.`) | ||||
| 		logger.Info(`Help us improve Traefik by leaving this feature on :)`) | ||||
| 		logger.Info(`More details on: https://docs.traefik.io/contributing/data-collection/`) | ||||
| 		logger.Info(`More details on: https://doc.traefik.io/traefik/contributing/data-collection/`) | ||||
| 		collect(staticConfiguration) | ||||
| 	} else { | ||||
| 		logger.Info(` | ||||
| Stats collection is disabled. | ||||
| Help us improve Traefik by turning this feature on :) | ||||
| More details on: https://docs.traefik.io/contributing/data-collection/ | ||||
| More details on: https://doc.traefik.io/traefik/contributing/data-collection/ | ||||
| `) | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										116
									
								
								cmd/traefik/traefik_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								cmd/traefik/traefik_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,116 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"crypto/x509" | ||||
| 	"encoding/pem" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/go-kit/kit/metrics" | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| 	"github.com/stretchr/testify/require" | ||||
| ) | ||||
|  | ||||
| // FooCert is a PEM-encoded TLS cert. | ||||
| // generated from src/crypto/tls: | ||||
| // go run generate_cert.go  --rsa-bits 1024 --host foo.org,foo.com  --ca --start-date "Jan 1 00:00:00 1970" --duration=1000000h | ||||
| const fooCert = `-----BEGIN CERTIFICATE----- | ||||
| MIICHzCCAYigAwIBAgIQXQFLeYRwc5X21t457t2xADANBgkqhkiG9w0BAQsFADAS | ||||
| MRAwDgYDVQQKEwdBY21lIENvMCAXDTcwMDEwMTAwMDAwMFoYDzIwODQwMTI5MTYw | ||||
| MDAwWjASMRAwDgYDVQQKEwdBY21lIENvMIGfMA0GCSqGSIb3DQEBAQUAA4GNADCB | ||||
| iQKBgQDCjn67GSs/khuGC4GNN+tVo1S+/eSHwr/hWzhfMqO7nYiXkFzmxi+u14CU | ||||
| Pda6WOeps7T2/oQEFMxKKg7zYOqkLSbjbE0ZfosopaTvEsZm/AZHAAvoOrAsIJOn | ||||
| SEiwy8h0tLA4z1SNR6rmIVQWyqBZEPAhBTQM1z7tFp48FakCFwIDAQABo3QwcjAO | ||||
| BgNVHQ8BAf8EBAMCAqQwEwYDVR0lBAwwCgYIKwYBBQUHAwEwDwYDVR0TAQH/BAUw | ||||
| AwEB/zAdBgNVHQ4EFgQUDHG3ASzeUezElup9zbPpBn/vjogwGwYDVR0RBBQwEoIH | ||||
| Zm9vLm9yZ4IHZm9vLmNvbTANBgkqhkiG9w0BAQsFAAOBgQBT+VLMbB9u27tBX8Aw | ||||
| ZrGY3rbNdBGhXVTksrjiF+6ZtDpD3iI56GH9zLxnqvXkgn3u0+Ard5TqF/xmdwVw | ||||
| NY0V/aWYfcL2G2auBCQrPvM03ozRnVUwVfP23eUzX2ORNHCYhd2ObQx4krrhs7cJ | ||||
| SWxtKwFlstoXY3K2g9oRD9UxdQ== | ||||
| -----END CERTIFICATE-----` | ||||
|  | ||||
| // BarCert is a PEM-encoded TLS cert. | ||||
| // generated from src/crypto/tls: | ||||
| // go run generate_cert.go  --rsa-bits 1024 --host bar.org,bar.com  --ca --start-date "Jan 1 00:00:00 1970" --duration=10000h | ||||
| const barCert = `-----BEGIN CERTIFICATE----- | ||||
| MIICHTCCAYagAwIBAgIQcuIcNEXzBHPoxna5S6wG4jANBgkqhkiG9w0BAQsFADAS | ||||
| MRAwDgYDVQQKEwdBY21lIENvMB4XDTcwMDEwMTAwMDAwMFoXDTcxMDIyMTE2MDAw | ||||
| MFowEjEQMA4GA1UEChMHQWNtZSBDbzCBnzANBgkqhkiG9w0BAQEFAAOBjQAwgYkC | ||||
| gYEAqtcrP+KA7D6NjyztGNIPMup9KiBMJ8QL+preog/YHR7SQLO3kGFhpS3WKMab | ||||
| SzMypC3ZX1PZjBP5ZzwaV3PFbuwlCkPlyxR2lOWmullgI7mjY0TBeYLDIclIzGRp | ||||
| mpSDDSpkW1ay2iJDSpXjlhmwZr84hrCU7BRTQJo91fdsRTsCAwEAAaN0MHIwDgYD | ||||
| VR0PAQH/BAQDAgKkMBMGA1UdJQQMMAoGCCsGAQUFBwMBMA8GA1UdEwEB/wQFMAMB | ||||
| Af8wHQYDVR0OBBYEFK8jnzFQvBAgWtfzOyXY4VSkwrTXMBsGA1UdEQQUMBKCB2Jh | ||||
| ci5vcmeCB2Jhci5jb20wDQYJKoZIhvcNAQELBQADgYEAJz0ifAExisC/ZSRhWuHz | ||||
| 7qs1i6Nd4+YgEVR8dR71MChP+AMxucY1/ajVjb9xlLys3GPE90TWSdVppabEVjZY | ||||
| Oq11nPKc50ItTt8dMku6t0JHBmzoGdkN0V4zJCBqdQJxhop8JpYJ0S9CW0eT93h3 | ||||
| ipYQSsmIINGtMXJ8VkP/MlM= | ||||
| -----END CERTIFICATE-----` | ||||
|  | ||||
| type gaugeMock struct { | ||||
| 	metrics map[string]float64 | ||||
| 	labels  string | ||||
| } | ||||
|  | ||||
| func (g gaugeMock) With(labelValues ...string) metrics.Gauge { | ||||
| 	g.labels = strings.Join(labelValues, ",") | ||||
| 	return g | ||||
| } | ||||
|  | ||||
| func (g gaugeMock) Set(value float64) { | ||||
| 	g.metrics[g.labels] = value | ||||
| } | ||||
|  | ||||
| func (g gaugeMock) Add(delta float64) { | ||||
| 	panic("implement me") | ||||
| } | ||||
|  | ||||
| func TestAppendCertMetric(t *testing.T) { | ||||
| 	testCases := []struct { | ||||
| 		desc     string | ||||
| 		certs    []string | ||||
| 		expected map[string]float64 | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc:     "No certs", | ||||
| 			certs:    []string{}, | ||||
| 			expected: map[string]float64{}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:  "One cert", | ||||
| 			certs: []string{fooCert}, | ||||
| 			expected: map[string]float64{ | ||||
| 				"cn,,serial,123624926713171615935660664614975025408,sans,foo.com,foo.org": 3.6e+09, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:  "Two certs", | ||||
| 			certs: []string{fooCert, barCert}, | ||||
| 			expected: map[string]float64{ | ||||
| 				"cn,,serial,123624926713171615935660664614975025408,sans,foo.com,foo.org": 3.6e+09, | ||||
| 				"cn,,serial,152706022658490889223053211416725817058,sans,bar.com,bar.org": 3.6e+07, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, test := range testCases { | ||||
| 		test := test | ||||
| 		t.Run(test.desc, func(t *testing.T) { | ||||
| 			t.Parallel() | ||||
|  | ||||
| 			gauge := &gaugeMock{ | ||||
| 				metrics: map[string]float64{}, | ||||
| 			} | ||||
|  | ||||
| 			for _, cert := range test.certs { | ||||
| 				block, _ := pem.Decode([]byte(cert)) | ||||
| 				parsedCert, err := x509.ParseCertificate(block.Bytes) | ||||
| 				require.NoError(t, err) | ||||
|  | ||||
| 				appendCertMetric(gauge, parsedCert) | ||||
| 			} | ||||
|  | ||||
| 			assert.Equal(t, test.expected, gauge.metrics) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
| @@ -7,8 +7,8 @@ import ( | ||||
| 	"runtime" | ||||
| 	"text/template" | ||||
|  | ||||
| 	"github.com/containous/traefik/v2/pkg/cli" | ||||
| 	"github.com/containous/traefik/v2/pkg/version" | ||||
| 	"github.com/traefik/paerser/cli" | ||||
| 	"github.com/traefik/traefik/v2/pkg/version" | ||||
| ) | ||||
|  | ||||
| var versionTemplate = `Version:      {{.Version}} | ||||
|   | ||||
| @@ -130,7 +130,7 @@ | ||||
|       "tableColumn": "", | ||||
|       "targets": [ | ||||
|         { | ||||
|           "expr": "count(kube_pod_status_ready{namespace=\"$namespace\",condition=\"true\",pod=~\"traefik.*\"})", | ||||
|           "expr": "count(kube_pod_status_ready{condition=\"true\",pod=~\"traefik.*\"})", | ||||
|           "format": "time_series", | ||||
|           "intervalFactor": 1, | ||||
|           "refId": "A" | ||||
| @@ -150,10 +150,7 @@ | ||||
|       "valueName": "current" | ||||
|     }, | ||||
|     { | ||||
|       "aliasColors": { | ||||
|         "Latency over 1 min": "rgb(9, 116, 190)", | ||||
|         "Latency over 5 min": "#bf1b00" | ||||
|       }, | ||||
|       "aliasColors": {}, | ||||
|       "bars": false, | ||||
|       "dashLength": 10, | ||||
|       "dashes": false, | ||||
| @@ -183,22 +180,17 @@ | ||||
|       "pointradius": 5, | ||||
|       "points": false, | ||||
|       "renderer": "flot", | ||||
|       "seriesOverrides": [ | ||||
|         { | ||||
|           "alias": "Latency over 5 min", | ||||
|           "yaxis": 1 | ||||
|         } | ||||
|       ], | ||||
|       "seriesOverrides": [], | ||||
|       "spaceLength": 10, | ||||
|       "stack": false, | ||||
|       "steppedLine": false, | ||||
|       "targets": [ | ||||
|         { | ||||
|           "expr": "histogram_quantile(0.$percentiles, sum(rate(traefik_entrypoint_request_duration_seconds_bucket{namespace=\"$namespace\", code=\"200\",method=\"GET\"}[5m])) by (le))", | ||||
|           "expr": "histogram_quantile(0.$percentiles, sum(rate(traefik_entrypoint_request_duration_seconds_bucket{code=~\"2..\"}[5m])) by (le))", | ||||
|           "format": "time_series", | ||||
|           "hide": false, | ||||
|           "intervalFactor": 1, | ||||
|           "legendFormat": "Latency over 1 min", | ||||
|           "legendFormat": "Latency over 5 min", | ||||
|           "refId": "A" | ||||
|         } | ||||
|       ], | ||||
| @@ -281,7 +273,7 @@ | ||||
|       "steppedLine": false, | ||||
|       "targets": [ | ||||
|         { | ||||
|           "expr": "histogram_quantile(0.$percentiles, rate(traefik_entrypoint_request_duration_seconds_bucket{namespace=\"$namespace\",code=\"200\",method=\"GET\"}[5m]))", | ||||
|           "expr": "histogram_quantile(0.$percentiles, sum(rate(traefik_entrypoint_request_duration_seconds_bucket{code=~\"2..\"}[5m])) by (instance, le))", | ||||
|           "format": "time_series", | ||||
|           "intervalFactor": 1, | ||||
|           "legendFormat": "{{ instance }}", | ||||
| @@ -343,7 +335,7 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 7, | ||||
|           "gridPos": { | ||||
|             "h": 7, | ||||
| @@ -379,7 +371,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(traefik_entrypoint_open_connections{namespace=\"$namespace\"}) by (method)", | ||||
|               "expr": "sum(traefik_entrypoint_open_connections) by (method)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "{{ method }}", | ||||
| @@ -431,7 +423,7 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 7, | ||||
| @@ -465,7 +457,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_entrypoint_request_duration_seconds_bucket{namespace=\"$namespace\",le=\"0.1\",code=\"200\"}[5m])) by (job) / sum(rate(traefik_entrypoint_request_duration_seconds_count{namespace=\"$namespace\",code=\"200\"}[5m])) by (job)", | ||||
|               "expr": "(sum(rate(traefik_entrypoint_request_duration_seconds_bucket{le=\"0.1\",code=\"200\"}[5m])) by (job) + sum(rate(traefik_entrypoint_request_duration_seconds_bucket{le=\"0.3\",code=\"200\"}[5m])) by (job)) / 2 / sum(rate(traefik_entrypoint_request_duration_seconds_count{code=\"200\"}[5m])) by (job)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Code 200", | ||||
| @@ -511,9 +503,97 @@ | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           "aliasColors": {}, | ||||
|           "bars": true, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 10, | ||||
|             "w": 24, | ||||
|             "x": 0, | ||||
|             "y": 23 | ||||
|           }, | ||||
|           "id": 3, | ||||
|           "legend": { | ||||
|             "alignAsTable": true, | ||||
|             "avg": true, | ||||
|             "current": true, | ||||
|             "max": true, | ||||
|             "min": false, | ||||
|             "rightSide": false, | ||||
|             "show": true, | ||||
|             "sort": "avg", | ||||
|             "sortDesc": true, | ||||
|             "total": false, | ||||
|             "values": true | ||||
|           }, | ||||
|           "lines": false, | ||||
|           "linewidth": 1, | ||||
|           "links": [], | ||||
|           "nullPointMode": "null", | ||||
|           "percentage": false, | ||||
|           "pointradius": 5, | ||||
|           "points": false, | ||||
|           "renderer": "flot", | ||||
|           "seriesOverrides": [], | ||||
|           "spaceLength": 10, | ||||
|           "stack": false, | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_entrypoint_requests_total[1m])) by (entrypoint)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{ entrypoint }}", | ||||
|               "refId": "A" | ||||
|             } | ||||
|           ], | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Service total requests over 1min per entrypoint", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
|             "value_type": "individual" | ||||
|           }, | ||||
|           "type": "graph", | ||||
|           "xaxis": { | ||||
|             "buckets": null, | ||||
|             "mode": "time", | ||||
|             "name": null, | ||||
|             "show": true, | ||||
|             "values": [] | ||||
|           }, | ||||
|           "yaxes": [ | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             }, | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             } | ||||
|           ], | ||||
|           "yaxis": { | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       "title": "Frontends (entrypoints)", | ||||
|       "title": "Entrypoints", | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
| @@ -522,7 +602,7 @@ | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 16 | ||||
|         "y": 33 | ||||
|       }, | ||||
|       "id": 24, | ||||
|       "panels": [ | ||||
| @@ -531,13 +611,13 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 7, | ||||
|           "gridPos": { | ||||
|             "h": 7, | ||||
|             "w": 12, | ||||
|             "x": 0, | ||||
|             "y": 17 | ||||
|             "y": 34 | ||||
|           }, | ||||
|           "id": 25, | ||||
|           "legend": { | ||||
| @@ -567,7 +647,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(traefik_backend_open_connections{namespace=\"$namespace\"}) by (method)", | ||||
|               "expr": "sum(traefik_service_open_connections) by (method)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "{{ method }}", | ||||
| @@ -619,13 +699,13 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 7, | ||||
|             "w": 12, | ||||
|             "x": 12, | ||||
|             "y": 17 | ||||
|             "y": 34 | ||||
|           }, | ||||
|           "id": 26, | ||||
|           "legend": { | ||||
| @@ -653,7 +733,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_request_duration_seconds_bucket{namespace=\"$namespace\",le=\"0.1\",code=\"200\"}[5m])) by (job) / sum(rate(traefik_backend_request_duration_seconds_count{namespace=\"$namespace\",code=\"200\"}[5m])) by (job)", | ||||
|               "expr": "(sum(rate(traefik_service_request_duration_seconds_bucket{le=\"0.1\",code=\"200\"}[5m])) by (job) + sum(rate(traefik_service_request_duration_seconds_bucket{le=\"0.3\",code=\"200\"}[5m])) by (job)) / 2 / sum(rate(traefik_service_request_duration_seconds_count{code=\"200\"}[5m])) by (job)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Code 200", | ||||
| @@ -699,9 +779,97 @@ | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           "aliasColors": {}, | ||||
|           "bars": true, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 10, | ||||
|             "w": 24, | ||||
|             "x": 0, | ||||
|             "y": 41 | ||||
|           }, | ||||
|           "id": 4, | ||||
|           "legend": { | ||||
|             "alignAsTable": true, | ||||
|             "avg": true, | ||||
|             "current": true, | ||||
|             "max": true, | ||||
|             "min": false, | ||||
|             "rightSide": false, | ||||
|             "show": true, | ||||
|             "sort": "avg", | ||||
|             "sortDesc": true, | ||||
|             "total": false, | ||||
|             "values": true | ||||
|           }, | ||||
|           "lines": false, | ||||
|           "linewidth": 1, | ||||
|           "links": [], | ||||
|           "nullPointMode": "null", | ||||
|           "percentage": false, | ||||
|           "pointradius": 5, | ||||
|           "points": false, | ||||
|           "renderer": "flot", | ||||
|           "seriesOverrides": [], | ||||
|           "spaceLength": 10, | ||||
|           "stack": false, | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_service_requests_total[1m])) by (service)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{ service }}", | ||||
|               "refId": "A" | ||||
|             } | ||||
|           ], | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Service total requests over 1min per service", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
|             "value_type": "individual" | ||||
|           }, | ||||
|           "type": "graph", | ||||
|           "xaxis": { | ||||
|             "buckets": null, | ||||
|             "mode": "time", | ||||
|             "name": null, | ||||
|             "show": true, | ||||
|             "values": [] | ||||
|           }, | ||||
|           "yaxes": [ | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             }, | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             } | ||||
|           ], | ||||
|           "yaxis": { | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       "title": "Backends", | ||||
|       "title": "Services", | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
| @@ -710,7 +878,7 @@ | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 17 | ||||
|         "y": 51 | ||||
|       }, | ||||
|       "id": 15, | ||||
|       "panels": [ | ||||
| @@ -725,7 +893,7 @@ | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 0, | ||||
|             "y": 18 | ||||
|             "y": 52 | ||||
|           }, | ||||
|           "id": 5, | ||||
|           "legend": { | ||||
| @@ -755,7 +923,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_requests_total{namespace=\"$namespace\",code=~\"2..\"}[5m])) by (method, code)", | ||||
|               "expr": "sum(rate(traefik_service_requests_total{code=~\"2..\"}[5m])) by (method, code)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{method}} : {{code}}", | ||||
| @@ -813,7 +981,7 @@ | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 12, | ||||
|             "y": 18 | ||||
|             "y": 52 | ||||
|           }, | ||||
|           "id": 27, | ||||
|           "legend": { | ||||
| @@ -841,7 +1009,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_requests_total{namespace=\"$namespace\",code=~\"5..\"}[5m])) by (method, code)", | ||||
|               "expr": "sum(rate(traefik_service_requests_total{code=~\"5..\"}[5m])) by (method, code)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{method}} : {{code}}", | ||||
| @@ -899,95 +1067,7 @@ | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 0, | ||||
|             "y": 27 | ||||
|           }, | ||||
|           "id": 3, | ||||
|           "legend": { | ||||
|             "alignAsTable": true, | ||||
|             "avg": true, | ||||
|             "current": true, | ||||
|             "max": true, | ||||
|             "min": false, | ||||
|             "rightSide": true, | ||||
|             "show": true, | ||||
|             "sort": "avg", | ||||
|             "sortDesc": true, | ||||
|             "total": false, | ||||
|             "values": true | ||||
|           }, | ||||
|           "lines": false, | ||||
|           "linewidth": 1, | ||||
|           "links": [], | ||||
|           "nullPointMode": "null", | ||||
|           "percentage": false, | ||||
|           "pointradius": 5, | ||||
|           "points": false, | ||||
|           "renderer": "flot", | ||||
|           "seriesOverrides": [], | ||||
|           "spaceLength": 10, | ||||
|           "stack": false, | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_requests_total{namespace=\"$namespace\"}[1m])) by (backend)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{ backend }}", | ||||
|               "refId": "A" | ||||
|             } | ||||
|           ], | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Backend total requests over 1min per backend", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
|             "value_type": "individual" | ||||
|           }, | ||||
|           "type": "graph", | ||||
|           "xaxis": { | ||||
|             "buckets": null, | ||||
|             "mode": "time", | ||||
|             "name": null, | ||||
|             "show": true, | ||||
|             "values": [] | ||||
|           }, | ||||
|           "yaxes": [ | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             }, | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             } | ||||
|           ], | ||||
|           "yaxis": { | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           "aliasColors": {}, | ||||
|           "bars": true, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 12, | ||||
|             "y": 27 | ||||
|             "y": 61 | ||||
|           }, | ||||
|           "id": 6, | ||||
|           "legend": { | ||||
| @@ -1016,7 +1096,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_requests_total{namespace=\"$namespace\",code!~\"2..|5..\"}[5m])) by (method, code)", | ||||
|               "expr": "sum(rate(traefik_service_requests_total{code!~\"2..|5..\"}[5m])) by (method, code)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{ method }} : {{code}}", | ||||
| @@ -1026,7 +1106,7 @@ | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Others status code over 5min", | ||||
|           "title": "Others statuses code over 5min", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
| @@ -1064,7 +1144,7 @@ | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       "title": "HTTP Codes  stats", | ||||
|       "title": "HTTP Codes stats", | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
| @@ -1073,7 +1153,7 @@ | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 18 | ||||
|         "y": 70 | ||||
|       }, | ||||
|       "id": 35, | ||||
|       "panels": [ | ||||
| @@ -1082,13 +1162,13 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 0, | ||||
|             "y": 19 | ||||
|             "y": 71 | ||||
|           }, | ||||
|           "id": 31, | ||||
|           "legend": { | ||||
| @@ -1116,21 +1196,21 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "max(container_memory_usage_bytes{namespace=\"$namespace\", container_name=\"traefik\"})", | ||||
|               "expr": "sum(container_memory_usage_bytes{container=\"traefik\"})", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Max memory used", | ||||
|               "legendFormat": "Memory used", | ||||
|               "refId": "A" | ||||
|             }, | ||||
|             { | ||||
|               "expr": "avg(kube_pod_container_resource_requests_memory_bytes{namespace=\"$namespace\", container=\"traefik\"})", | ||||
|               "expr": "sum(kube_pod_container_resource_requests_memory_bytes{container=\"traefik\"})", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Requested memory usage", | ||||
|               "legendFormat": "Requested memory", | ||||
|               "refId": "B" | ||||
|             }, | ||||
|             { | ||||
|               "expr": "avg(kube_pod_container_resource_limits_memory_bytes{namespace=\"$namespace\", container=\"traefik\"})", | ||||
|               "expr": "sum(kube_pod_container_resource_limits_memory_bytes{container=\"traefik\"})", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Limit memory usage", | ||||
| @@ -1140,7 +1220,7 @@ | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Traefik max memory usage", | ||||
|           "title": "Traefik memory usage", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
| @@ -1182,13 +1262,13 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 12, | ||||
|             "y": 19 | ||||
|             "y": 71 | ||||
|           }, | ||||
|           "id": 33, | ||||
|           "legend": { | ||||
| @@ -1215,21 +1295,21 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "max(rate(container_cpu_usage_seconds_total{namespace=\"$namespace\", container_name=\"traefik\"}[1m]))", | ||||
|               "expr": "sum(rate(container_cpu_usage_seconds_total{container=\"traefik\"}[2m]))", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Max cpu used", | ||||
|               "legendFormat": "Cpu used", | ||||
|               "refId": "A" | ||||
|             }, | ||||
|             { | ||||
|               "expr": "avg(kube_pod_container_resource_requests_cpu_cores{namespace=\"$namespace\", container=\"traefik\"})", | ||||
|               "expr": "sum(kube_pod_container_resource_requests_cpu_cores{container=\"traefik\"})", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Requested cpu usage", | ||||
|               "legendFormat": "Requested cpu", | ||||
|               "refId": "B" | ||||
|             }, | ||||
|             { | ||||
|               "expr": "avg(kube_pod_container_resource_limits_cpu_cores{namespace=\"$namespace\", container=\"traefik\"})", | ||||
|               "expr": "sum(kube_pod_container_resource_limits_cpu_cores{container=\"traefik\"})", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Limit cpu usage", | ||||
| @@ -1239,7 +1319,7 @@ | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Traefik max CPU usage", | ||||
|           "title": "Traefik CPU usage", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
| @@ -1277,7 +1357,7 @@ | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       "title": "Pods ressources", | ||||
|       "title": "Pods resources", | ||||
|       "type": "row" | ||||
|     } | ||||
|   ], | ||||
| @@ -1288,26 +1368,6 @@ | ||||
|   ], | ||||
|   "templating": { | ||||
|     "list": [ | ||||
|       { | ||||
|         "allValue": null, | ||||
|         "current": {}, | ||||
|         "datasource": "${DS_PROMETHEUS}", | ||||
|         "hide": 0, | ||||
|         "includeAll": false, | ||||
|         "label": null, | ||||
|         "multi": false, | ||||
|         "name": "namespace", | ||||
|         "options": [], | ||||
|         "query": "label_values(traefik_config_reloads_total, namespace)", | ||||
|         "refresh": 1, | ||||
|         "regex": "", | ||||
|         "sort": 0, | ||||
|         "tagValuesQuery": "", | ||||
|         "tags": [], | ||||
|         "tagsQuery": "", | ||||
|         "type": "query", | ||||
|         "useTags": false | ||||
|       }, | ||||
|       { | ||||
|         "allValue": null, | ||||
|         "current": { | ||||
| @@ -1370,5 +1430,5 @@ | ||||
|   "timezone": "", | ||||
|   "title": "Traefik", | ||||
|   "uid": "traefik-kubernetes", | ||||
|   "version": 1 | ||||
|   "version": 2 | ||||
| } | ||||
|   | ||||
| @@ -64,10 +64,7 @@ | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
|       "aliasColors": { | ||||
|         "Latency over 1 min": "rgb(9, 116, 190)", | ||||
|         "Latency over 5 min": "#bf1b00" | ||||
|       }, | ||||
|       "aliasColors": {}, | ||||
|       "bars": false, | ||||
|       "dashLength": 10, | ||||
|       "dashes": false, | ||||
| @@ -97,22 +94,17 @@ | ||||
|       "pointradius": 5, | ||||
|       "points": false, | ||||
|       "renderer": "flot", | ||||
|       "seriesOverrides": [ | ||||
|         { | ||||
|           "alias": "Latency over 5 min", | ||||
|           "yaxis": 1 | ||||
|         } | ||||
|       ], | ||||
|       "seriesOverrides": [], | ||||
|       "spaceLength": 10, | ||||
|       "stack": false, | ||||
|       "steppedLine": false, | ||||
|       "targets": [ | ||||
|         { | ||||
|           "expr": "histogram_quantile(0.$percentiles, sum(rate(traefik_entrypoint_request_duration_seconds_bucket{code=\"200\",method=\"GET\"}[5m])) by (le))", | ||||
|           "expr": "histogram_quantile(0.$percentiles, sum(rate(traefik_entrypoint_request_duration_seconds_bucket{code=~\"2..\"}[5m])) by (le))", | ||||
|           "format": "time_series", | ||||
|           "hide": false, | ||||
|           "intervalFactor": 1, | ||||
|           "legendFormat": "Latency over 1 min", | ||||
|           "legendFormat": "Latency over 5 min", | ||||
|           "refId": "A" | ||||
|         } | ||||
|       ], | ||||
| @@ -195,7 +187,7 @@ | ||||
|       "steppedLine": false, | ||||
|       "targets": [ | ||||
|         { | ||||
|           "expr": "histogram_quantile(0.$percentiles, rate(traefik_entrypoint_request_duration_seconds_bucket{code=\"200\",method=\"GET\"}[5m]))", | ||||
|           "expr": "histogram_quantile(0.$percentiles, sum(rate(traefik_entrypoint_request_duration_seconds_bucket{code=~\"2..\"}[5m])) by (instance, le))", | ||||
|           "format": "time_series", | ||||
|           "intervalFactor": 1, | ||||
|           "legendFormat": "{{ instance }}", | ||||
| @@ -257,13 +249,13 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 7, | ||||
|           "gridPos": { | ||||
|             "h": 7, | ||||
|             "w": 12, | ||||
|             "x": 0, | ||||
|             "y": 2 | ||||
|             "y": 16 | ||||
|           }, | ||||
|           "id": 19, | ||||
|           "legend": { | ||||
| @@ -345,13 +337,13 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 7, | ||||
|             "w": 12, | ||||
|             "x": 12, | ||||
|             "y": 2 | ||||
|             "y": 16 | ||||
|           }, | ||||
|           "id": 22, | ||||
|           "legend": { | ||||
| @@ -379,7 +371,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_entrypoint_request_duration_seconds_bucket{le=\"0.1\",code=\"200\"}[5m])) by (job) / sum(rate(traefik_entrypoint_request_duration_seconds_count{code=\"200\"}[5m])) by (job)", | ||||
|               "expr": "(sum(rate(traefik_entrypoint_request_duration_seconds_bucket{le=\"0.1\",code=\"200\"}[5m])) by (job) + sum(rate(traefik_entrypoint_request_duration_seconds_bucket{le=\"0.3\",code=\"200\"}[5m])) by (job)) / 2 / sum(rate(traefik_entrypoint_request_duration_seconds_count{code=\"200\"}[5m])) by (job)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Code 200", | ||||
| @@ -425,9 +417,97 @@ | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           "aliasColors": {}, | ||||
|           "bars": true, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 10, | ||||
|             "w": 24, | ||||
|             "x": 0, | ||||
|             "y": 23 | ||||
|           }, | ||||
|           "id": 3, | ||||
|           "legend": { | ||||
|             "alignAsTable": true, | ||||
|             "avg": true, | ||||
|             "current": true, | ||||
|             "max": true, | ||||
|             "min": false, | ||||
|             "rightSide": false, | ||||
|             "show": true, | ||||
|             "sort": "avg", | ||||
|             "sortDesc": true, | ||||
|             "total": false, | ||||
|             "values": true | ||||
|           }, | ||||
|           "lines": false, | ||||
|           "linewidth": 1, | ||||
|           "links": [], | ||||
|           "nullPointMode": "null", | ||||
|           "percentage": false, | ||||
|           "pointradius": 5, | ||||
|           "points": false, | ||||
|           "renderer": "flot", | ||||
|           "seriesOverrides": [], | ||||
|           "spaceLength": 10, | ||||
|           "stack": false, | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_entrypoint_requests_total[1m])) by (entrypoint)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{ entrypoint }}", | ||||
|               "refId": "A" | ||||
|             } | ||||
|           ], | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Service total requests over 1min per entrypoint", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
|             "value_type": "individual" | ||||
|           }, | ||||
|           "type": "graph", | ||||
|           "xaxis": { | ||||
|             "buckets": null, | ||||
|             "mode": "time", | ||||
|             "name": null, | ||||
|             "show": true, | ||||
|             "values": [] | ||||
|           }, | ||||
|           "yaxes": [ | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             }, | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             } | ||||
|           ], | ||||
|           "yaxis": { | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       "title": "Frontends (entrypoints)", | ||||
|       "title": "Entrypoints", | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
| @@ -436,7 +516,7 @@ | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 16 | ||||
|         "y": 33 | ||||
|       }, | ||||
|       "id": 24, | ||||
|       "panels": [ | ||||
| @@ -445,13 +525,13 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 7, | ||||
|           "gridPos": { | ||||
|             "h": 7, | ||||
|             "w": 12, | ||||
|             "x": 0, | ||||
|             "y": 3 | ||||
|             "y": 34 | ||||
|           }, | ||||
|           "id": 25, | ||||
|           "legend": { | ||||
| @@ -481,7 +561,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(traefik_backend_open_connections) by (method)", | ||||
|               "expr": "sum(traefik_service_open_connections) by (method)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "{{ method }}", | ||||
| @@ -533,13 +613,13 @@ | ||||
|           "bars": false, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": null, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 7, | ||||
|             "w": 12, | ||||
|             "x": 12, | ||||
|             "y": 3 | ||||
|             "y": 34 | ||||
|           }, | ||||
|           "id": 26, | ||||
|           "legend": { | ||||
| @@ -567,7 +647,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_request_duration_seconds_bucket{le=\"0.1\",code=\"200\"}[5m])) by (job) / sum(rate(traefik_backend_request_duration_seconds_count{code=\"200\"}[5m])) by (job)", | ||||
|               "expr": "(sum(rate(traefik_service_request_duration_seconds_bucket{le=\"0.1\",code=\"200\"}[5m])) by (job) + sum(rate(traefik_service_request_duration_seconds_bucket{le=\"0.3\",code=\"200\"}[5m])) by (job)) / 2 / sum(rate(traefik_service_request_duration_seconds_count{code=\"200\"}[5m])) by (job)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 1, | ||||
|               "legendFormat": "Code 200", | ||||
| @@ -613,9 +693,97 @@ | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           "aliasColors": {}, | ||||
|           "bars": true, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 10, | ||||
|             "w": 24, | ||||
|             "x": 0, | ||||
|             "y": 41 | ||||
|           }, | ||||
|           "id": 4, | ||||
|           "legend": { | ||||
|             "alignAsTable": true, | ||||
|             "avg": true, | ||||
|             "current": true, | ||||
|             "max": true, | ||||
|             "min": false, | ||||
|             "rightSide": false, | ||||
|             "show": true, | ||||
|             "sort": "avg", | ||||
|             "sortDesc": true, | ||||
|             "total": false, | ||||
|             "values": true | ||||
|           }, | ||||
|           "lines": false, | ||||
|           "linewidth": 1, | ||||
|           "links": [], | ||||
|           "nullPointMode": "null", | ||||
|           "percentage": false, | ||||
|           "pointradius": 5, | ||||
|           "points": false, | ||||
|           "renderer": "flot", | ||||
|           "seriesOverrides": [], | ||||
|           "spaceLength": 10, | ||||
|           "stack": false, | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_service_requests_total[1m])) by (service)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{ service }}", | ||||
|               "refId": "A" | ||||
|             } | ||||
|           ], | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Service total requests over 1min per service", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
|             "value_type": "individual" | ||||
|           }, | ||||
|           "type": "graph", | ||||
|           "xaxis": { | ||||
|             "buckets": null, | ||||
|             "mode": "time", | ||||
|             "name": null, | ||||
|             "show": true, | ||||
|             "values": [] | ||||
|           }, | ||||
|           "yaxes": [ | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             }, | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             } | ||||
|           ], | ||||
|           "yaxis": { | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       "title": "Backends", | ||||
|       "title": "Services", | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
| @@ -624,7 +792,7 @@ | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 17 | ||||
|         "y": 51 | ||||
|       }, | ||||
|       "id": 15, | ||||
|       "panels": [ | ||||
| @@ -639,7 +807,7 @@ | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 0, | ||||
|             "y": 4 | ||||
|             "y": 52 | ||||
|           }, | ||||
|           "id": 5, | ||||
|           "legend": { | ||||
| @@ -669,7 +837,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_requests_total{code=~\"2..\"}[5m])) by (method, code)", | ||||
|               "expr": "sum(rate(traefik_service_requests_total{code=~\"2..\"}[5m])) by (method, code)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{method}} : {{code}}", | ||||
| @@ -727,7 +895,7 @@ | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 12, | ||||
|             "y": 4 | ||||
|             "y": 52 | ||||
|           }, | ||||
|           "id": 27, | ||||
|           "legend": { | ||||
| @@ -755,7 +923,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_requests_total{code=~\"5..\"}[5m])) by (method, code)", | ||||
|               "expr": "sum(rate(traefik_service_requests_total{code=~\"5..\"}[5m])) by (method, code)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{method}} : {{code}}", | ||||
| @@ -813,95 +981,7 @@ | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 0, | ||||
|             "y": 13 | ||||
|           }, | ||||
|           "id": 3, | ||||
|           "legend": { | ||||
|             "alignAsTable": true, | ||||
|             "avg": true, | ||||
|             "current": true, | ||||
|             "max": true, | ||||
|             "min": false, | ||||
|             "rightSide": true, | ||||
|             "show": true, | ||||
|             "sort": "avg", | ||||
|             "sortDesc": true, | ||||
|             "total": false, | ||||
|             "values": true | ||||
|           }, | ||||
|           "lines": false, | ||||
|           "linewidth": 1, | ||||
|           "links": [], | ||||
|           "nullPointMode": "null", | ||||
|           "percentage": false, | ||||
|           "pointradius": 5, | ||||
|           "points": false, | ||||
|           "renderer": "flot", | ||||
|           "seriesOverrides": [], | ||||
|           "spaceLength": 10, | ||||
|           "stack": false, | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_requests_total[1m])) by (backend)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{ backend }}", | ||||
|               "refId": "A" | ||||
|             } | ||||
|           ], | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Backend total requests over 1min per backend", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
|             "value_type": "individual" | ||||
|           }, | ||||
|           "type": "graph", | ||||
|           "xaxis": { | ||||
|             "buckets": null, | ||||
|             "mode": "time", | ||||
|             "name": null, | ||||
|             "show": true, | ||||
|             "values": [] | ||||
|           }, | ||||
|           "yaxes": [ | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             }, | ||||
|             { | ||||
|               "format": "short", | ||||
|               "label": null, | ||||
|               "logBase": 1, | ||||
|               "max": null, | ||||
|               "min": null, | ||||
|               "show": true | ||||
|             } | ||||
|           ], | ||||
|           "yaxis": { | ||||
|             "align": false, | ||||
|             "alignLevel": null | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           "aliasColors": {}, | ||||
|           "bars": true, | ||||
|           "dashLength": 10, | ||||
|           "dashes": false, | ||||
|           "datasource": "${DS_PROMETHEUS}", | ||||
|           "fill": 1, | ||||
|           "gridPos": { | ||||
|             "h": 9, | ||||
|             "w": 12, | ||||
|             "x": 12, | ||||
|             "y": 13 | ||||
|             "y": 61 | ||||
|           }, | ||||
|           "id": 6, | ||||
|           "legend": { | ||||
| @@ -930,7 +1010,7 @@ | ||||
|           "steppedLine": false, | ||||
|           "targets": [ | ||||
|             { | ||||
|               "expr": "sum(rate(traefik_backend_requests_total{code!~\"2..|5..\"}[5m])) by (method, code)", | ||||
|               "expr": "sum(rate(traefik_service_requests_total{code!~\"2..|5..\"}[5m])) by (method, code)", | ||||
|               "format": "time_series", | ||||
|               "intervalFactor": 2, | ||||
|               "legendFormat": "{{ method }} : {{code}}", | ||||
| @@ -940,7 +1020,7 @@ | ||||
|           "thresholds": [], | ||||
|           "timeFrom": null, | ||||
|           "timeShift": null, | ||||
|           "title": "Others status code over 5min", | ||||
|           "title": "Others statuses code over 5min", | ||||
|           "tooltip": { | ||||
|             "shared": true, | ||||
|             "sort": 0, | ||||
| @@ -978,7 +1058,7 @@ | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       "title": "HTTP Codes  stats", | ||||
|       "title": "HTTP Codes stats", | ||||
|       "type": "row" | ||||
|     } | ||||
|   ], | ||||
| @@ -1051,5 +1131,5 @@ | ||||
|   "timezone": "", | ||||
|   "title": "Traefik", | ||||
|   "uid": "traefik", | ||||
|   "version": 1 | ||||
|   "version": 2 | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| [Unit] | ||||
| Description=Traefik | ||||
| Documentation=https://docs.traefik.io | ||||
| Documentation=https://doc.traefik.io/traefik/ | ||||
| #After=network-online.target | ||||
| #AssertFileIsExecutable=/usr/bin/traefik | ||||
| #AssertPathExists=/etc/traefik/traefik.toml | ||||
|   | ||||
							
								
								
									
										10
									
								
								debug.Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								debug.Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| FROM alpine:3.14 | ||||
| # Feel free to add below any helpful dependency for debugging. | ||||
| # iproute2 is for ss. | ||||
| RUN apk --no-cache --no-progress add bash curl ca-certificates tzdata lsof iproute2 \ | ||||
|     && update-ca-certificates \ | ||||
|     && rm -rf /var/cache/apk/* | ||||
| COPY dist/traefik / | ||||
| EXPOSE 80 | ||||
| VOLUME ["/tmp"] | ||||
| ENTRYPOINT ["/traefik"] | ||||
| @@ -4,6 +4,7 @@ | ||||
|     "MD009": false, | ||||
|     "MD013": false, | ||||
|     "MD024": false, | ||||
|     "MD025": false, | ||||
|     "MD026": false, | ||||
|     "MD033": false, | ||||
|     "MD034": false, | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
|  | ||||
| ####### | ||||
| # This Makefile contains all targets related to the documentation | ||||
| ####### | ||||
| @@ -12,41 +11,55 @@ TRAEFIK_DOCS_CHECK_IMAGE ?= $(TRAEFIK_DOCS_BUILD_IMAGE)-check | ||||
| SITE_DIR := $(CURDIR)/site | ||||
|  | ||||
| DOCKER_RUN_DOC_PORT := 8000 | ||||
| DOCKER_RUN_DOC_MOUNTS := -v $(CURDIR):/mkdocs  | ||||
| DOCKER_RUN_DOC_MOUNTS := -v $(CURDIR):/mkdocs | ||||
| DOCKER_RUN_DOC_OPTS := --rm $(DOCKER_RUN_DOC_MOUNTS) -p $(DOCKER_RUN_DOC_PORT):8000 | ||||
|  | ||||
| # Default: generates the documentation into $(SITE_DIR) | ||||
| .PHONY: docs | ||||
| docs: docs-clean docs-image docs-lint docs-build docs-verify | ||||
|  | ||||
| # Writer Mode: build and serve docs on http://localhost:8000 with livereload | ||||
| .PHONY: docs-serve | ||||
| docs-serve: docs-image | ||||
| 	docker run  $(DOCKER_RUN_DOC_OPTS) $(TRAEFIK_DOCS_BUILD_IMAGE) mkdocs serve | ||||
|  | ||||
| ## Pull image for doc building | ||||
| .PHONY: docs-pull-images | ||||
| docs-pull-images: | ||||
| 	grep --no-filename -E '^FROM' ./*.Dockerfile \ | ||||
| 		| awk '{print $$2}' \ | ||||
| 		| sort \ | ||||
| 		| uniq \ | ||||
| 		| xargs -P 6 -n 1 docker pull | ||||
|  | ||||
| # Utilities Targets for each step | ||||
| .PHONY: docs-image | ||||
| docs-image: | ||||
| 	docker build -t $(TRAEFIK_DOCS_BUILD_IMAGE) -f docs.Dockerfile ./ | ||||
|  | ||||
| .PHONY: docs-build | ||||
| docs-build: docs-image | ||||
| 	docker run $(DOCKER_RUN_DOC_OPTS) $(TRAEFIK_DOCS_BUILD_IMAGE) sh -c "mkdocs build \ | ||||
| 		&& chown -R $(shell id -u):$(shell id -g) ./site" | ||||
|  | ||||
| .PHONY: docs-verify | ||||
| docs-verify: docs-build | ||||
| 	@if [ "$(DOCS_VERIFY_SKIP)" != "true" ]; then \ | ||||
| 		docker build -t $(TRAEFIK_DOCS_CHECK_IMAGE) -f check.Dockerfile ./; \ | ||||
| 		docker run --rm -v $(CURDIR):/app $(TRAEFIK_DOCS_CHECK_IMAGE) /verify.sh; \ | ||||
| 	else \ | ||||
| 		echo "DOCS_VERIFY_SKIP is true: no verification done."; \ | ||||
| 	fi | ||||
| ifneq ("$(DOCS_VERIFY_SKIP)", "true") | ||||
| 	docker build -t $(TRAEFIK_DOCS_CHECK_IMAGE) -f check.Dockerfile ./ | ||||
| 	docker run --rm -v $(CURDIR):/app $(TRAEFIK_DOCS_CHECK_IMAGE) /verify.sh | ||||
| else | ||||
| 	echo "DOCS_VERIFY_SKIP is true: no verification done." | ||||
| endif | ||||
|  | ||||
| .PHONY: docs-lint | ||||
| docs-lint: | ||||
| 	@if [ "$(DOCS_LINT_SKIP)" != "true" ]; then \ | ||||
| 		docker build -t $(TRAEFIK_DOCS_CHECK_IMAGE) -f check.Dockerfile ./ && \ | ||||
| 		docker run --rm -v $(CURDIR):/app $(TRAEFIK_DOCS_CHECK_IMAGE) /lint.sh; \ | ||||
| 	else \ | ||||
| 		echo "DOCS_LINT_SKIP is true: no linting done."; \ | ||||
| 	fi | ||||
| ifneq ("$(DOCS_LINT_SKIP)", "true") | ||||
| 	docker build -t $(TRAEFIK_DOCS_CHECK_IMAGE) -f check.Dockerfile ./ | ||||
| 	docker run --rm -v $(CURDIR):/app $(TRAEFIK_DOCS_CHECK_IMAGE) /lint.sh | ||||
| else | ||||
| 	echo "DOCS_LINT_SKIP is true: no linting done." | ||||
| endif | ||||
|  | ||||
| .PHONY: docs-clean | ||||
| docs-clean: | ||||
| 	rm -rf $(SITE_DIR) | ||||
|  | ||||
| .PHONY: all docs-verify docs docs-clean docs-build docs-lint | ||||
|   | ||||
| @@ -1,15 +1,21 @@ | ||||
|  | ||||
| FROM alpine:3.10 as alpine | ||||
| FROM alpine:3.18 as alpine | ||||
|  | ||||
| RUN apk --no-cache --no-progress add \ | ||||
|     build-base \ | ||||
|     gcompat \ | ||||
|     libcurl \ | ||||
|     libxml2-dev \ | ||||
|     libxslt-dev \ | ||||
|     ruby \ | ||||
|     ruby-bigdecimal \ | ||||
|     ruby-dev \ | ||||
|     ruby-etc \ | ||||
|     ruby-ffi \ | ||||
|     ruby-json \ | ||||
|     ruby-nokogiri | ||||
| RUN gem install html-proofer --version 3.13.0 --no-document -- --use-system-libraries | ||||
|     zlib-dev | ||||
|  | ||||
| RUN gem install nokogiri --version 1.15.3 --no-document -- --use-system-libraries | ||||
| RUN gem install html-proofer --version 5.0.7 --no-document -- --use-system-libraries | ||||
|  | ||||
| # After Ruby, some NodeJS YAY! | ||||
| RUN apk --no-cache --no-progress add \ | ||||
| @@ -17,12 +23,9 @@ RUN apk --no-cache --no-progress add \ | ||||
|     nodejs \ | ||||
|     npm | ||||
|  | ||||
| # To handle 'not get uid/gid' | ||||
| RUN npm config set unsafe-perm true | ||||
|  | ||||
| RUN npm install --global \ | ||||
|     markdownlint@0.17.2 \ | ||||
|     markdownlint-cli@0.19.0 | ||||
|     markdownlint@0.29.0 \ | ||||
|     markdownlint-cli@0.35.0 | ||||
|  | ||||
| # Finally the shell tools we need for later | ||||
| # tini helps to terminate properly all the parallelized tasks when sending CTRL-C | ||||
|   | ||||
| @@ -1 +0,0 @@ | ||||
| docs.traefik.io | ||||
							
								
								
									
										
											BIN
										
									
								
								docs/content/assets/img/maintainers-guidelines.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/content/assets/img/maintainers-guidelines.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 966 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/content/assets/img/providers/nomad.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/content/assets/img/providers/nomad.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 13 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 2.0 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/content/assets/img/traefik.logo-dark.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/content/assets/img/traefik.logo-dark.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 38 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/content/assets/img/traefikproxy-icon-color.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/content/assets/img/traefikproxy-icon-color.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 7.6 KiB | 
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| After Width: | Height: | Size: 20 KiB | 
| @@ -1,96 +0,0 @@ | ||||
| /* | ||||
|  | ||||
| Atom One Light by Daniel Gamage | ||||
| Original One Light Syntax theme from https://github.com/atom/one-light-syntax | ||||
|  | ||||
| base:    #fafafa | ||||
| mono-1:  #383a42 | ||||
| mono-2:  #686b77 | ||||
| mono-3:  #a0a1a7 | ||||
| hue-1:   #0184bb | ||||
| hue-2:   #4078f2 | ||||
| hue-3:   #a626a4 | ||||
| hue-4:   #50a14f | ||||
| hue-5:   #e45649 | ||||
| hue-5-2: #c91243 | ||||
| hue-6:   #986801 | ||||
| hue-6-2: #c18401 | ||||
|  | ||||
| */ | ||||
|  | ||||
| .hljs { | ||||
|   display: block; | ||||
|   overflow-x: auto; | ||||
|   padding: 0.5em; | ||||
|   color: #383a42; | ||||
|   background: #fafafa; | ||||
| } | ||||
|  | ||||
| .hljs-comment, | ||||
| .hljs-quote { | ||||
|   color: #a0a1a7; | ||||
|   font-style: italic; | ||||
| } | ||||
|  | ||||
| .hljs-doctag, | ||||
| .hljs-keyword, | ||||
| .hljs-formula { | ||||
|   color: #a626a4; | ||||
| } | ||||
|  | ||||
| .hljs-section, | ||||
| .hljs-name, | ||||
| .hljs-selector-tag, | ||||
| .hljs-deletion, | ||||
| .hljs-subst { | ||||
|   color: #e45649; | ||||
| } | ||||
|  | ||||
| .hljs-literal { | ||||
|   color: #0184bb; | ||||
| } | ||||
|  | ||||
| .hljs-string, | ||||
| .hljs-regexp, | ||||
| .hljs-addition, | ||||
| .hljs-attribute, | ||||
| .hljs-meta-string { | ||||
|   color: #50a14f; | ||||
| } | ||||
|  | ||||
| .hljs-built_in, | ||||
| .hljs-class .hljs-title { | ||||
|   color: #c18401; | ||||
| } | ||||
|  | ||||
| .hljs-attr, | ||||
| .hljs-variable, | ||||
| .hljs-template-variable, | ||||
| .hljs-type, | ||||
| .hljs-selector-class, | ||||
| .hljs-selector-attr, | ||||
| .hljs-selector-pseudo, | ||||
| .hljs-number { | ||||
|   color: #986801; | ||||
| } | ||||
|  | ||||
| .hljs-symbol, | ||||
| .hljs-bullet, | ||||
| .hljs-link, | ||||
| .hljs-meta, | ||||
| .hljs-selector-id, | ||||
| .hljs-title { | ||||
|   color: #4078f2; | ||||
| } | ||||
|  | ||||
| .hljs-emphasis { | ||||
|   font-style: italic; | ||||
| } | ||||
|  | ||||
| .hljs-strong { | ||||
|   font-weight: bold; | ||||
| } | ||||
|  | ||||
| .hljs-link { | ||||
|   text-decoration: underline; | ||||
| } | ||||
| @@ -1,63 +0,0 @@ | ||||
| @import url('https://fonts.googleapis.com/css?family=Noto+Sans|Noto+Serif'); | ||||
|  | ||||
| .md-logo img { | ||||
|     background-color: white; | ||||
|     border-radius: 50%; | ||||
|     width: 30px; | ||||
|     height: 30px; | ||||
| } | ||||
|  | ||||
| /* Fix for Chrome */ | ||||
| .md-typeset__table td code { | ||||
|     word-break: unset; | ||||
| } | ||||
|  | ||||
| .md-typeset__table tr :nth-child(1) { | ||||
|     word-wrap: break-word; | ||||
|     max-width: 30em; | ||||
| } | ||||
|  | ||||
| body { | ||||
|     font-family: 'Noto Sans', sans-serif; | ||||
| } | ||||
|  | ||||
| h1 { | ||||
|     font-weight: bold !important; | ||||
|     color: rgba(0,0,0,.9) !important; | ||||
| } | ||||
|  | ||||
| h2 { | ||||
|     font-weight: bold !important; | ||||
| } | ||||
|  | ||||
| h3 { | ||||
|     font-weight: bold !important; | ||||
| } | ||||
|  | ||||
| .md-typeset h5 { | ||||
|     text-transform: none; | ||||
| } | ||||
|  | ||||
| figcaption { | ||||
|     text-align: center; | ||||
|     font-size: 0.8em; | ||||
|     font-style: italic; | ||||
|     color: #8D909F; | ||||
| } | ||||
|  | ||||
| p.subtitle { | ||||
|     color: rgba(0,0,0,.54); | ||||
|     padding-top: 0; | ||||
|     margin-top: -2em; | ||||
|     font-weight: bold; | ||||
|     font-size: 1.25em; | ||||
| } | ||||
|  | ||||
| .markdown-body .task-list-item { | ||||
|     list-style-type: none !important; | ||||
| } | ||||
|  | ||||
| .markdown-body .task-list-item input[type="checkbox"] { | ||||
|     margin: 0 4px 0.25em -20px; | ||||
|     vertical-align: middle; | ||||
| } | ||||
| @@ -1,10 +1,31 @@ | ||||
| --- | ||||
| title: "Traefik Advocation Documentation" | ||||
| description: "There are many ways to contribute to Traefik Proxy. Let us know if you’re talking about Traefik, and we'll promote your enthusiasm!" | ||||
| --- | ||||
|  | ||||
| # Advocating | ||||
|  | ||||
| Spread the Love & Tell Us about It | ||||
| Spread the Love & Tell Us About It | ||||
| {: .subtitle } | ||||
|  | ||||
| There are many ways to contribute to the project, and there is one that always spark joy: when we see/read about users talking about how Traefik helps them solve their problems. | ||||
| Traefik Proxy was started by the community and for the community. | ||||
| You can contribute to the Traefik community in three main ways: | ||||
|  | ||||
| If you're talking about Traefik, [let us know](https://blog.containo.us/spread-the-love-ba5a40aa72e7) and we'll promote your enthusiasm! | ||||
| **Spread the word!** Guides, videos, blog posts, how-to articles, and showing off your network design all help spread the word about Traefik Proxy | ||||
| and teach others in the community how to best implement it. | ||||
| It always sparks joy when users share how Traefik Proxy helps them solve their problems. | ||||
| If you are talking about Traefik Proxy, [let us know](https://traefik.io/submit-my-contribution/) and we will promote your work and reward your enthusiasm! | ||||
| If you are giving a talk that includes or is about Traefik Proxy, [let us know](https://traefik.io/submit-my-contribution/) and we will send you swag and stickers for your time at the conference. | ||||
| If you have written about Traefik or shared useful information you would like to promote, feel free to add links to the [dedicated wiki page on GitHub](https://github.com/traefik/traefik/wiki/Awesome-Traefik). | ||||
|  | ||||
| Also, if you've written about Traefik or shared useful information you'd like to promote, feel free to add links in the [dedicated wiki page on Github](https://github.com/containous/traefik/wiki/Awesome-Traefik). | ||||
| **Help community members!** Everyone needs a place to share their cool innovations or get help with that pesky bug that only a different pair of eyes seems to be able to see. | ||||
| Join our [Community Forum](https://community.traefik.io/) where you can ask questions, help out other users, and share your neat configuration examples or snippets. | ||||
| Top contributors will be asked to join the Ambassador program and get unique swag to celebrate! | ||||
|  | ||||
| **Build cool solutions!** Traefik Proxy would be so much better if only it had… | ||||
| We love all the wonderful ideas that our users come up with, but we can only build so much. | ||||
| Luckily, as an open source community, our users can help by [building awesome features](https://github.com/orgs/traefik/projects/9/views/7), enhancements, or bug fixes. | ||||
| We are a big community, so we do need to prioritize a bit. | ||||
| That is why we use the tag `contributor/wanted` to let you know which pull requests will make it to the front of the queue for design support and review. | ||||
| Feel free to grab one of these and run with it. | ||||
| Top contributors get unique swag to celebrate. | ||||
|   | ||||
| @@ -1,19 +1,29 @@ | ||||
| --- | ||||
| title: "Traefik Building & Testing Documentation" | ||||
| description: "Compile and test your own Traefik Proxy! Learn how to build your own Traefik binary from the sources, and read the technical documentation." | ||||
| --- | ||||
|  | ||||
| # Building and Testing | ||||
|  | ||||
| Compile and Test Your Own Traefik! | ||||
| {: .subtitle } | ||||
|  | ||||
| So you want to build your own Traefik binary from the sources? | ||||
| You want to build your own Traefik binary from the sources? | ||||
| Let's see how. | ||||
|  | ||||
| ## Building | ||||
|  | ||||
| You need either [Docker](https://github.com/docker/docker) and `make` (Method 1), or `go` (Method 2) in order to build Traefik. | ||||
| You need either [Docker](https://github.com/docker/docker "Link to website of Docker") and `make` (Method 1), or [Go](https://go.dev/ "Link to website of Go") (Method 2) in order to build Traefik. | ||||
| For changes to its dependencies, the `dep` dependency management tool is required. | ||||
|  | ||||
| ### Method 1: Using `Docker` and `Makefile` | ||||
|  | ||||
| Run make with the `binary` target. | ||||
|  | ||||
| ```bash | ||||
| make binary | ||||
| ``` | ||||
|  | ||||
| This will create binaries for the Linux platform in the `dist` folder. | ||||
|  | ||||
| In case when you run build on CI, you may probably want to run docker in non-interactive mode. To achieve that define `DOCKER_NON_INTERACTIVE=true` environment variable. | ||||
| @@ -30,12 +40,12 @@ Successfully tagged traefik-webui:latest | ||||
| [...] | ||||
| docker build  -t "traefik-dev:4475--feature-documentation" -f build.Dockerfile . | ||||
| Sending build context to Docker daemon    279MB | ||||
| Step 1/10 : FROM golang:1.14-alpine | ||||
| Step 1/10 : FROM golang:1.16-alpine | ||||
|  ---> f4bfb3d22bda | ||||
| [...] | ||||
| Successfully built 5c3c1a911277 | ||||
| Successfully tagged traefik-dev:4475--feature-documentation | ||||
| docker run  -e "TEST_CONTAINER=1" -v "/var/run/docker.sock:/var/run/docker.sock" -it -e OS_ARCH_ARG -e OS_PLATFORM_ARG -e TESTFLAGS -e VERBOSE -e VERSION -e CODENAME -e TESTDIRS -e CI -e CONTAINER=DOCKER		 -v "/home/ldez/sources/go/src/github.com/containous/traefik/"dist":/go/src/github.com/containous/traefik/"dist"" "traefik-dev:4475--feature-documentation" ./script/make.sh generate binary | ||||
| docker run  -e "TEST_CONTAINER=1" -v "/var/run/docker.sock:/var/run/docker.sock" -it -e OS_ARCH_ARG -e OS_PLATFORM_ARG -e TESTFLAGS -e VERBOSE -e VERSION -e CODENAME -e TESTDIRS -e CI -e CONTAINER=DOCKER		 -v "/home/ldez/sources/go/src/github.com/traefik/traefik/"dist":/go/src/github.com/traefik/traefik/"dist"" "traefik-dev:4475--feature-documentation" ./script/make.sh generate binary | ||||
| ---> Making bundle: generate (in .) | ||||
| removed 'autogen/genstatic/gen.go' | ||||
|  | ||||
| @@ -45,7 +55,7 @@ $ ls dist/ | ||||
| traefik* | ||||
| ``` | ||||
|  | ||||
| The following targets can be executed outside Docker by setting the variable `PRE_TARGET` to an empty string (we don't recommend that): | ||||
| The following targets can be executed outside Docker by setting the variable `IN_DOCKER` to an empty string (although be aware that some of the tests might fail in that context): | ||||
|  | ||||
| - `test-unit` | ||||
| - `test-integration` | ||||
| @@ -55,20 +65,19 @@ The following targets can be executed outside Docker by setting the variable `PR | ||||
| ex: | ||||
|  | ||||
| ```bash | ||||
| PRE_TARGET= make test-unit | ||||
| IN_DOCKER= make test-unit | ||||
| ``` | ||||
|  | ||||
| ### Method 2: Using `go` | ||||
|  | ||||
| Requirements: | ||||
|  | ||||
| - `go` v1.14+ | ||||
| - `go` v1.16+ | ||||
| - environment variable `GO111MODULE=on` | ||||
| - [go-bindata](https://github.com/containous/go-bindata) `GO111MODULE=off go get -u github.com/containous/go-bindata/...` | ||||
|  | ||||
| !!! tip "Source Directory" | ||||
|  | ||||
|     It is recommended that you clone Traefik into the `~/go/src/github.com/containous/traefik` directory. | ||||
|     It is recommended that you clone Traefik into the `~/go/src/github.com/traefik/traefik` directory. | ||||
|     This is the official golang workspace hierarchy that will allow dependencies to be properly resolved. | ||||
|  | ||||
| !!! note "Environment" | ||||
| @@ -101,18 +110,9 @@ Requirements: | ||||
|  | ||||
| Once you've set up your go environment and cloned the source repository, you can build Traefik. | ||||
|  | ||||
| Beforehand, you need to get [go-bindata](https://github.com/containous/go-bindata) (the first time) in order to be able to use the `go generate` command (which is part of the build process). | ||||
|  | ||||
| ```bash | ||||
| cd ~/go/src/github.com/containous/traefik | ||||
|  | ||||
| # Get go-bindata. (Important: the ellipses are required.) | ||||
| GO111MODULE=off go get github.com/containous/go-bindata/... | ||||
| ``` | ||||
|  | ||||
| ```bash | ||||
| # Generate UI static files | ||||
| rm -rf static/ autogen/; make generate-webui | ||||
| make clean-webui generate-webui | ||||
|  | ||||
| # required to merge non-code components into the final binary, | ||||
| # such as the web dashboard/UI | ||||
| @@ -124,7 +124,7 @@ go generate | ||||
| go build ./cmd/traefik | ||||
| ``` | ||||
|  | ||||
| You will find the Traefik executable (`traefik`) in the `~/go/src/github.com/containous/traefik` directory. | ||||
| You will find the Traefik executable (`traefik`) in the `~/go/src/github.com/traefik/traefik` directory. | ||||
|  | ||||
| ## Testing | ||||
|  | ||||
| @@ -138,13 +138,13 @@ Run all tests (unit and integration) using the `test` target. | ||||
| $ make test-unit | ||||
| docker build -t "traefik-dev:your-feature-branch" -f build.Dockerfile . | ||||
| # […] | ||||
| docker run --rm -it -e OS_ARCH_ARG -e OS_PLATFORM_ARG -e TESTFLAGS -v "/home/user/go/src/github/containous/traefik/dist:/go/src/github.com/containous/traefik/dist" "traefik-dev:your-feature-branch" ./script/make.sh generate test-unit | ||||
| docker run --rm -it -e OS_ARCH_ARG -e OS_PLATFORM_ARG -e TESTFLAGS -v "/home/user/go/src/github/traefik/traefik/dist:/go/src/github.com/traefik/traefik/dist" "traefik-dev:your-feature-branch" ./script/make.sh generate test-unit | ||||
| ---> Making bundle: generate (in .) | ||||
| removed 'gen.go' | ||||
|  | ||||
| ---> Making bundle: test-unit (in .) | ||||
| + go test -cover -coverprofile=cover.out . | ||||
| ok      github.com/containous/traefik   0.005s  coverage: 4.1% of statements | ||||
| ok      github.com/traefik/traefik   0.005s  coverage: 4.1% of statements | ||||
|  | ||||
| Test success | ||||
| ``` | ||||
| @@ -165,14 +165,14 @@ TESTFLAGS="-check.f MyTestSuite.My" make test-integration | ||||
| TESTFLAGS="-check.f MyTestSuite.*Test" make test-integration | ||||
| ``` | ||||
|  | ||||
| More: https://labix.org/gocheck | ||||
| Check [gocheck](https://labix.org/gocheck "Link to website of gocheck") for more information. | ||||
|  | ||||
| ### Method 2: `go` | ||||
|  | ||||
| Unit tests can be run from the cloned directory using `$ go test ./...` which should return `ok`, similar to: | ||||
|  | ||||
| ```test | ||||
| ok      _/home/user/go/src/github/containous/traefik    0.004s | ||||
| ok      _/home/user/go/src/github/traefik/traefik    0.004s | ||||
| ``` | ||||
|  | ||||
| Integration tests must be run from the `integration/` directory and require the `-integration` switch: `$ cd integration && go test -integration ./...`. | ||||
|   | ||||
| @@ -1,3 +1,8 @@ | ||||
| --- | ||||
| title: "Traefik Data Collection Documentation" | ||||
| description: "To learn more about how Traefik is being used and improve it, we collect anonymous usage statistics from running instances. Read the technical documentation." | ||||
| --- | ||||
|  | ||||
| # Data Collection | ||||
|  | ||||
| Understanding How Traefik is Being Used | ||||
| @@ -5,23 +10,23 @@ Understanding How Traefik is Being Used | ||||
|  | ||||
| ## Configuration Example | ||||
|  | ||||
| Understanding how you use Traefik is very important to us: it helps us improve the solution in many different ways.   | ||||
| For this very reason, the sendAnonymousUsage option is mandatory: we want you to take time to consider whether or not you wish to share anonymous data with us so we can benefit from your experience and use cases. | ||||
| Understanding how you use Traefik is very important to us: it helps us improve the solution in many different ways. | ||||
| For this very reason, the sendAnonymousUsage option is mandatory: we want you to take time to consider whether or not you wish to share anonymous data with us, so we can benefit from your experience and use cases. | ||||
|  | ||||
| !!! example "Enabling Data Collection" | ||||
|      | ||||
|     ```toml tab="File (TOML)" | ||||
|     [global] | ||||
|       # Send anonymous usage data | ||||
|       sendAnonymousUsage = true | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```yaml tab="File (YAML)" | ||||
|     global: | ||||
|       # Send anonymous usage data | ||||
|       sendAnonymousUsage: true | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [global] | ||||
|       # Send anonymous usage data | ||||
|       sendAnonymousUsage = true | ||||
|     ``` | ||||
|  | ||||
|     ```bash tab="CLI" | ||||
|     # Send anonymous usage data | ||||
|     --global.sendAnonymousUsage | ||||
| @@ -29,7 +34,7 @@ For this very reason, the sendAnonymousUsage option is mandatory: we want you to | ||||
|  | ||||
| ## Collected Data | ||||
|  | ||||
| This feature comes from the public proposal [here](https://github.com/containous/traefik/issues/2369). | ||||
| This feature comes from this [public proposal](https://github.com/traefik/traefik/issues/2369). | ||||
|  | ||||
| In order to help us learn more about how Traefik is being used and improve it, we collect anonymous usage statistics from running instances. | ||||
| Those data help us prioritize our developments and focus on what's important for our users (for example, which provider is popular, and which is not). | ||||
| @@ -40,56 +45,58 @@ Once a day (the first call begins 10 minutes after the start of Traefik), we col | ||||
|  | ||||
| - the Traefik version number | ||||
| - a hash of the configuration | ||||
| - an **anonymized version** of the static configuration (token, user name, password, URL, IP, domain, email, etc, are removed). | ||||
| - an **anonymized version** of the static configuration (token, username, password, URL, IP, domain, email, etc., are removed). | ||||
|  | ||||
| !!! info | ||||
|      | ||||
|  | ||||
|     - We do not collect the dynamic configuration information (routers & services). | ||||
|     - We do not collect this data to run advertising programs. | ||||
|     - We do not sell this data to third-parties. | ||||
|  | ||||
| ### Example of Collected Data | ||||
|  | ||||
| ```toml tab="Original configuration" | ||||
| [entryPoints] | ||||
|   [entryPoints.web] | ||||
|     address = ":80" | ||||
| ```yaml tab="Original configuration" | ||||
| entryPoints: | ||||
|   web: | ||||
|   address: ":80" | ||||
|  | ||||
| [api] | ||||
| api: {} | ||||
|  | ||||
| [providers.docker] | ||||
|   endpoint = "tcp://10.10.10.10:2375" | ||||
|   exposedByDefault = true | ||||
|   swarmMode = true | ||||
| providers: | ||||
|   docker: | ||||
|     endpoint: "tcp://10.10.10.10:2375" | ||||
|     exposedByDefault: true | ||||
|     swarmMode: true | ||||
|  | ||||
|   [providers.docker.TLS] | ||||
|     ca = "dockerCA" | ||||
|     cert = "dockerCert" | ||||
|     key = "dockerKey" | ||||
|     insecureSkipVerify = true | ||||
|     tls: | ||||
|       ca: dockerCA | ||||
|       cert: dockerCert | ||||
|       key: dockerKey | ||||
|       insecureSkipVerify: true | ||||
| ``` | ||||
|  | ||||
| ```toml tab="Resulting Obfuscated Configuration" | ||||
| [entryPoints] | ||||
|   [entryPoints.web] | ||||
|     address = ":80" | ||||
| ```yaml tab="Resulting Obfuscated Configuration" | ||||
| entryPoints: | ||||
|   web: | ||||
|   address: ":80" | ||||
|  | ||||
| [api] | ||||
| api: {} | ||||
|  | ||||
| [providers.docker] | ||||
|   endpoint = "xxxx" | ||||
|   exposedByDefault = true | ||||
|   swarmMode = true | ||||
| providers: | ||||
|   docker: | ||||
|     endpoint: "xxxx" | ||||
|     exposedByDefault: true | ||||
|     swarmMode: true | ||||
|  | ||||
|   [providers.docker.TLS] | ||||
|     ca = "xxxx" | ||||
|     cert = "xxxx" | ||||
|     key = "xxxx" | ||||
|     insecureSkipVerify = false | ||||
|     tls: | ||||
|       ca: xxxx | ||||
|       cert: xxxx | ||||
|       key: xxxx | ||||
|       insecureSkipVerify: true | ||||
| ``` | ||||
|  | ||||
| ## The Code for Data Collection | ||||
|  | ||||
| If you want to dig into more details, here is the source code of the collecting system: [collector.go](https://github.com/containous/traefik/blob/master/pkg/collector/collector.go) | ||||
| If you want to dig into more details, here is the source code of the collecting system: [collector.go](https://github.com/traefik/traefik/blob/master/pkg/collector/collector.go) | ||||
|  | ||||
| By default we anonymize all configuration fields, except fields tagged with `export=true`. | ||||
| By default, we anonymize all configuration fields, except fields tagged with `export=true`. | ||||
|   | ||||
| @@ -1,3 +1,8 @@ | ||||
| --- | ||||
| title: "Traefik Contribution Documentation" | ||||
| description: "Found something unclear in the Traefik Proxy documentation and want to give a try at explaining it better? Read the guide to building documentation." | ||||
| --- | ||||
|  | ||||
| # Documentation | ||||
|  | ||||
| Features Are Better When You Know How to Use Them | ||||
| @@ -10,17 +15,21 @@ Let's see how. | ||||
|  | ||||
| ### General | ||||
|  | ||||
| This [documentation](https://docs.traefik.io/) is built with [mkdocs](https://mkdocs.org/). | ||||
| This [documentation](https://doc.traefik.io/traefik/ "Link to the official Traefik documentation") is built with [MkDocs](https://mkdocs.org/ "Link to website of MkDocs"). | ||||
|  | ||||
| ### Method 1: `Docker` and `make` | ||||
|  | ||||
| You can build the documentation and test it locally (with live reloading), using the `docs` target: | ||||
| Please make sure you have the following requirements installed: | ||||
|  | ||||
| - [Docker](https://www.docker.com/ "Link to website of Docker") | ||||
|  | ||||
| You can build the documentation and test it locally (with live reloading), using the `docs-serve` target: | ||||
|  | ||||
| ```bash | ||||
| $ make docs | ||||
| $ make docs-serve | ||||
| docker build -t traefik-docs -f docs.Dockerfile . | ||||
| # […] | ||||
| docker run  --rm -v /home/user/go/github/containous/traefik:/mkdocs -p 8000:8000 traefik-docs mkdocs serve | ||||
| docker run  --rm -v /home/user/go/github/traefik/traefik:/mkdocs -p 8000:8000 traefik-docs mkdocs serve | ||||
| # […] | ||||
| [I 170828 20:47:48 server:283] Serving on http://0.0.0.0:8000 | ||||
| [I 170828 20:47:48 handlers:60] Start watching changes | ||||
| @@ -29,7 +38,7 @@ docker run  --rm -v /home/user/go/github/containous/traefik:/mkdocs -p 8000:8000 | ||||
|  | ||||
| !!! tip "Default URL" | ||||
|  | ||||
|     Your local documentation server will run by default on [http://127.0.0.1:8000](http://127.0.0.1:8000). | ||||
|     Your local documentation server will run by default on <http://127.0.0.1:8000>. | ||||
|  | ||||
| If you only want to build the documentation without serving it locally, you can use the following command: | ||||
|  | ||||
| @@ -38,9 +47,12 @@ $ make docs-build | ||||
| ... | ||||
| ``` | ||||
|  | ||||
| ### Method 2: `mkdocs` | ||||
| ### Method 2: `MkDocs` | ||||
|  | ||||
| First, make sure you have `python` and `pip` installed. | ||||
| Please make sure you have the following requirements installed: | ||||
|  | ||||
| - [Python](https://www.python.org/ "Link to website of Python") | ||||
| - [pip](https://pypi.org/project/pip/ "Link to the website of pip on PyPI") | ||||
|  | ||||
| ```bash | ||||
| $ python --version | ||||
| @@ -49,7 +61,7 @@ $ pip --version | ||||
| pip 1.5.2 | ||||
| ``` | ||||
|  | ||||
| Then, install mkdocs with `pip`. | ||||
| Then, install MkDocs with `pip`. | ||||
|  | ||||
| ```bash | ||||
| pip install --user -r requirements.txt | ||||
| @@ -75,24 +87,26 @@ To check that the documentation meets standard expectations (no dead links, html | ||||
| $ make docs-verify | ||||
| docker build -t traefik-docs-verify ./script/docs-verify-docker-image ## Build Validator image | ||||
| ... | ||||
| docker run --rm -v /home/travis/build/containous/traefik:/app traefik-docs-verify ## Check for dead links and w3c compliance | ||||
| docker run --rm -v /home/travis/build/traefik/traefik:/app traefik-docs-verify ## Check for dead links and w3c compliance | ||||
| === Checking HTML content... | ||||
| Running ["HtmlCheck", "ImageCheck", "ScriptCheck", "LinkCheck"] on /app/site/basics/index.html on *.html... | ||||
| ``` | ||||
|  | ||||
| !!! note "Clean & Verify" | ||||
|  | ||||
|     If you've made changes to the documentation, it's safter to clean it before verifying it.  | ||||
|     If you've made changes to the documentation, it's safer to clean it before verifying it. | ||||
|  | ||||
|     ```bash | ||||
|     $ make docs-clean docs-verify | ||||
|     $ make docs | ||||
|     ... | ||||
|     ``` | ||||
|  | ||||
|     Will perform all necessary steps for you. | ||||
|  | ||||
| !!! note "Disabling Documentation Verification" | ||||
|  | ||||
|     Verification can be disabled by setting the environment variable `DOCS_VERIFY_SKIP` to `true`: | ||||
|      | ||||
|  | ||||
|     ```shell | ||||
|     DOCS_VERIFY_SKIP=true make docs-verify | ||||
|     ... | ||||
|   | ||||
							
								
								
									
										127
									
								
								docs/content/contributing/maintainers-guidelines.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								docs/content/contributing/maintainers-guidelines.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,127 @@ | ||||
| --- | ||||
| title: "Traefik Maintainer's Guidelines Documentation" | ||||
| description: "Interested in contributing more to the community and becoming a Traefik Proxy maintainer? Read the guide to becoming a part of the core team." | ||||
| --- | ||||
|  | ||||
| # Maintainer's Guidelines | ||||
|  | ||||
|  | ||||
|  | ||||
| Welcome to the Traefik Community. | ||||
|  | ||||
| We are strongly promoting a philosophy of openness and sharing, | ||||
| and firmly standing against the elitist closed approach. | ||||
| Being part of the core team should be accessible to anyone motivated | ||||
| and wants to be part of that journey! | ||||
|  | ||||
| ## Becoming a Maintainer | ||||
|  | ||||
| Before a contributor becomes a maintainer, they should meet the following requirements: | ||||
|  | ||||
| - The contributor enabled [2FA](https://docs.github.com/en/authentication/securing-your-account-with-two-factor-authentication-2fa/configuring-two-factor-authentication) on their GitHub account | ||||
|  | ||||
| - The contributor showed a consistent pattern of helpful, non-threatening, and friendly behavior towards other community members in the past. | ||||
|  | ||||
| - The contributor has read and accepted the maintainer's guidelines. | ||||
|  | ||||
| The contributor should also meet one or several of the following requirements: | ||||
|  | ||||
| - The contributor has opened and successfully run medium to large PR’s in the past 6 months. | ||||
|  | ||||
| - The contributor has participated in multiple code reviews of other PR’s, | ||||
|   including those of other maintainers and contributors. | ||||
|  | ||||
| - The contributor is active on Traefik Community forums | ||||
|   or other technical forums/boards such as K8S slack, Reddit, StackOverflow, hacker news. | ||||
|  | ||||
| Any existing active maintainer can create an issue to discuss promoting a contributor to maintainer.  | ||||
| Other maintainers can vote on the issue, and if the quorum is reached, the contributor is promoted to maintainer. | ||||
| If the quorum is not reached within one month after the issue is created, it is closed. | ||||
|  | ||||
| ## Maintainer's Responsibilities and Privileges | ||||
|  | ||||
| As a maintainer, you are granted a vote for the following: | ||||
|  | ||||
| - [PR review](https://github.com/traefik/contributors-guide/blob/master/pr_guidelines.md). | ||||
|  | ||||
| - [Design review](https://github.com/traefik/contributors-guide/blob/master/proposals.md). | ||||
|  | ||||
| - [Proposals](https://github.com/traefik/contributors-guide/blob/master/proposals.md). | ||||
|  | ||||
| Maintainers are also added to the maintainer's Discord server where happens the [issue triage](https://github.com/traefik/contributors-guide/blob/master/issue_triage.md) | ||||
| and appear on the [Maintainers](maintainers.md) page. | ||||
|  | ||||
| As a maintainer, you should:  | ||||
|  | ||||
| - Prioritize PR reviews, design reviews, and issue triage above any other task.  | ||||
|  | ||||
| Making sure contributors and community members are listened to and have an impact on the project is essential to keeping the project active and develop a thriving community. | ||||
|  | ||||
| - Prioritize helping contributors reaching the expecting quality level over rewriting contributions. | ||||
|  | ||||
| Any triage activity on issues and PRs (e.g. labels, marking messages as off-topic, refusing, marking duplicates) should result from a collective decision to ensure knowledge is shared among maintainers. | ||||
|  | ||||
| ## Communicating | ||||
|  | ||||
| - All of our maintainers are added to the Traefik Maintainers Discord server that belongs to Traefik labs. | ||||
|   Having the team in one place helps us to communicate effectively. | ||||
|   Maintainers can discuss issues, pull requests, enhancements more efficiently | ||||
|   and get the feedback almost immediately. | ||||
|   Fewer blockers mean more fun and engaging work. | ||||
|  | ||||
| - Every decision made on the discord server among maintainers is documented so it's visible to the rest of the community. | ||||
|  | ||||
| - Maintainers express their opinions on issues and reviews.  | ||||
|   It is fine to have different point of views.  | ||||
|   We encourage active and open conversations which goals are to improve Traefik. | ||||
|  | ||||
| - When discussing issues and proposals, maintainers should share as much information as possible to help solve the issue. | ||||
|  | ||||
| ## Maintainers Activity | ||||
|  | ||||
| In order to keep the core team efficient and dynamic, | ||||
| maintainers' activity and involvement will be reviewed on a regular basis. | ||||
|  | ||||
| - Has the maintainer engaged with the team and the community by meeting two or more of these benchmarks in the past six months? | ||||
|  | ||||
|     - Has the maintainer participated in at least two or three maintainer meetings? | ||||
|  | ||||
|     - Substantial review of at least one or two PRs from either contributors or maintainers. | ||||
|  | ||||
|     - Opened at least one or two bug fixes or feature request PRs | ||||
|       that were eventually merged (or on a trajectory for merge). | ||||
|  | ||||
|     - Substantial participation in the Help Wanted program (answered questions, helped identify issues, applied guidelines from the Help Wanted guide to open issues). | ||||
|  | ||||
|     - Substantial participation with the community in general. | ||||
|  | ||||
| - Has the maintainer shown a consistent pattern of helpful, | ||||
|   non-threatening, | ||||
|   and friendly behavior towards other people on the maintainer team and with our community? | ||||
|  | ||||
| ## Additional Comments for Maintainers (that should apply to any contributor) | ||||
|  | ||||
| - Be respectful with other maintainers and other community members. | ||||
|  | ||||
| - Be open minded when participating in conversations: try to put yourself in others’ shoes. | ||||
|  | ||||
| - Keep the communication public - | ||||
|   if anyone tries to communicate with you directly, | ||||
|   ask politely to move the conversation to a public communication channel. | ||||
|  | ||||
| - Stay away from defensive comments. | ||||
|  | ||||
| - Please try to express your thoughts clearly enough | ||||
|   and note that some of us are not native English speakers. | ||||
|   Try to rephrase your sentences, avoiding mental shortcuts; | ||||
|   none of us is able to predict anyone's thoughts. | ||||
|  | ||||
| - Be proactive. | ||||
|  | ||||
| - Emoji are fine, | ||||
|   but if you express yourself clearly enough they are not necessary. | ||||
|   They will not replace good communication. | ||||
|  | ||||
| - Embrace mentorship: help others grow and match the quality level we strive for. | ||||
|  | ||||
| - Keep in mind that we all have the same goal: improve the project. | ||||
| @@ -1,176 +1,37 @@ | ||||
| --- | ||||
| title: "Traefik Maintainers Documentation" | ||||
| description: "Traefik Proxy is an open source software with a thriving community of contributors and maintainers. Read the list of maintainers on this page." | ||||
| --- | ||||
|  | ||||
| # Maintainers | ||||
|  | ||||
| ## The team | ||||
| ## Active Maintainers | ||||
|  | ||||
| * Emile Vauge [@emilevauge](https://github.com/emilevauge) | ||||
| * Vincent Demeester [@vdemeester](https://github.com/vdemeester) | ||||
| * Ed Robinson [@errm](https://github.com/errm) | ||||
| * Daniel Tomcej [@dtomcej](https://github.com/dtomcej) | ||||
| * Manuel Zapf [@SantoDE](https://github.com/SantoDE) | ||||
| * Timo Reimann [@timoreimann](https://github.com/timoreimann) | ||||
| * Ludovic Fernandez [@ldez](https://github.com/ldez) | ||||
| * Julien Salleyron [@juliens](https://github.com/juliens) | ||||
| * Nicolas Mengin [@nmengin](https://github.com/nmengin) | ||||
| * Marco Jantke [@marco-jantke](https://github.com/marco-jantke) | ||||
| * Michaël Matur [@mmatur](https://github.com/mmatur) | ||||
| * Gérald Croës [@geraldcroes](https://github.com/geraldcroes) | ||||
| * Jean-Baptiste Doumenjou [@jbdoumenjou](https://github.com/jbdoumenjou) | ||||
| * Mathieu Lonjaret [@mpl](https://github.com/mpl) | ||||
| * Romain Tribotté [@rtribotte](https://github.com/rtribotte) | ||||
| * Kevin Pollet [@kevinpollet](https://github.com/kevinpollet) | ||||
| * Harold Ozouf [@jspdown](https://github.com/jspdown) | ||||
| * Tom Moulard [@tommoulard](https://github.com/tommoulard) | ||||
| * Landry Benguigui [@lbenguigui](https://github.com/lbenguigui) | ||||
|  | ||||
| ## Contributions Daily Meeting | ||||
| ## Past Maintainers | ||||
|  | ||||
| * 3 Maintainers should attend to a Contributions Daily Meeting where we sort and label new issues ([is:issue label:status/0-needs-triage](https://github.com/containous/traefik/issues?utf8=%E2%9C%93&q=is%3Aissue+label%3Astatus%2F0-needs-triage+)), and review every Pull Requests | ||||
| * Every pull request should be checked during the Contributions Daily Meeting | ||||
|     * Even if it’s already assigned | ||||
|     * Even PR labelled with `contributor/waiting-for-corrections` or `contributor/waiting-for-feedback` | ||||
| * Issues labeled with `priority/P0` and `priority/P1` should be assigned. | ||||
| * Modifying an issue or a pull request (labels, assignees, milestone) is only possible: | ||||
|     * During the Contributions Daily Meeting | ||||
|     * By an assigned maintainer | ||||
|     * In case of emergency, if a change proposal is approved by 2 other maintainers (on Slack, Discord, Discourse, etc) | ||||
| People who have had an incredibly positive impact on the project, and are now focusing on other projects. | ||||
|  | ||||
| ## PR review process: | ||||
| * Vincent Demeester [@vdemeester](https://github.com/vdemeester) | ||||
| * Ed Robinson [@errm](https://github.com/errm) | ||||
| * Daniel Tomcej [@dtomcej](https://github.com/dtomcej) | ||||
| * Timo Reimann [@timoreimann](https://github.com/timoreimann) | ||||
| * Marco Jantke [@mjantke](https://github.com/mjeri) | ||||
|  | ||||
| * The status `needs-design-review` is only used in complex/heavy/tricky PRs. | ||||
| * From `1` to `2`: 1 comment that says “design LGTM” (by a senior maintainer). | ||||
| * From `2` to `3`: 3 LGTM approvals by any maintainer. | ||||
| * If needed, a specific maintainer familiar with a particular domain can be requested for the review. | ||||
| * If a PR has been implemented in pair programming, one peer's LGTM goes into the review for free | ||||
| * Amending someone else's pull request is authorized only in emergency, if a rebase is needed, or if the initial contributor is silent | ||||
| ## Maintainer's Guidelines | ||||
|  | ||||
| We use [PRM](https://github.com/ldez/prm) to manage locally pull requests. | ||||
|  | ||||
| ## Bots | ||||
|  | ||||
| ### [Myrmica Lobicornis](https://github.com/containous/lobicornis/) | ||||
|  | ||||
| Update and Merge Pull Request. | ||||
|  | ||||
| The maintainer giving the final LGTM must add the `status/3-needs-merge` label to trigger the merge bot. | ||||
|  | ||||
| By default, a squash-rebase merge will be carried out. | ||||
| To preserve commits, add `bot/merge-method-rebase` before `status/3-needs-merge`. | ||||
|  | ||||
| The status `status/4-merge-in-progress` is only used by the bot. | ||||
|  | ||||
| If the bot is not able to perform the merge, the label `bot/need-human-merge` is added.   | ||||
| In such a situation, solve the conflicts/CI/... and then remove the label `bot/need-human-merge`. | ||||
|  | ||||
| To prevent the bot from automatically merging a PR, add the label `bot/no-merge`. | ||||
|  | ||||
| The label `bot/light-review` decreases the number of required LGTM from 3 to 1. | ||||
|  | ||||
| This label is used when: | ||||
|  | ||||
| * Updating the vendors from previously reviewed PRs | ||||
| * Merging branches into the master | ||||
| * Preparing the release | ||||
|  | ||||
| ### [Myrmica Bibikoffi](https://github.com/containous/bibikoffi/) | ||||
|  | ||||
| * closes stale issues [cron] | ||||
|     * use some criterion as number of days between creation, last update, labels, ... | ||||
|  | ||||
| ### [Myrmica Aloba](https://github.com/containous/aloba) | ||||
|  | ||||
| Manage GitHub labels. | ||||
|  | ||||
| * Add labels on new PR [GitHub WebHook] | ||||
| * Add milestone to a new PR based on a branch version (1.4, 1.3, ...) [GitHub WebHook] | ||||
| * Add and remove `contributor/waiting-for-corrections` label when a review request changes [GitHub WebHook] | ||||
| * Weekly report of PR status on Slack (CaptainPR) [cron] | ||||
|  | ||||
| ## Labels | ||||
|  | ||||
| A maintainer that looks at an issue/PR must define its `kind/*`, `area/*`, and `status/*`. | ||||
|  | ||||
| ### Status - Workflow | ||||
|  | ||||
| The `status/*` labels represent the desired state in the workflow. | ||||
|  | ||||
| * `status/0-needs-triage`: all the new issues and PRs have this status. _[bot only]_ | ||||
| * `status/1-needs-design-review`: needs a design review. **(only for PR)** | ||||
| * `status/2-needs-review`: needs a code/documentation review. **(only for PR)** | ||||
| * `status/3-needs-merge`: ready to merge. **(only for PR)** | ||||
| * `status/4-merge-in-progress`: merge is in progress. _[bot only]_ | ||||
|  | ||||
| ### Contributor | ||||
|  | ||||
| * `contributor/need-more-information`: we need more information from the contributor in order to analyze a problem. | ||||
| * `contributor/waiting-for-feedback`: we need the contributor to give us feedback. | ||||
| * `contributor/waiting-for-corrections`: we need the contributor to take actions in order to move forward with a PR. **(only for PR)** _[bot, humans]_ | ||||
| * `contributor/needs-resolve-conflicts`: use it only when there is some conflicts (and an automatic rebase is not possible). **(only for PR)** _[bot, humans]_ | ||||
|  | ||||
| ### Kind | ||||
|  | ||||
| * `kind/enhancement`: a new or improved feature. | ||||
| * `kind/question`: a question. **(only for issue)** | ||||
| * `kind/proposal`: a proposal that needs to be discussed. | ||||
|     * _Proposal issues_ are design proposals | ||||
|     * _Proposal PRs_ are technical prototypes that need to be refined with multiple contributors. | ||||
|  | ||||
| * `kind/bug/possible`: a possible bug that needs analysis before it is confirmed or fixed. **(only for issues)** | ||||
| * `kind/bug/confirmed`: a confirmed bug (reproducible). **(only for issues)** | ||||
| * `kind/bug/fix`: a bug fix. **(only for PR)** | ||||
|  | ||||
| ### Resolution | ||||
|  | ||||
| * `resolution/duplicate`: a duplicate issue/PR. | ||||
| * `resolution/declined`: declined (Rule #1 of open-source: no is temporary, yes is forever). | ||||
| * `WIP`: Work In Progress. **(only for PR)** | ||||
|  | ||||
| ### Platform | ||||
|  | ||||
| * `platform/windows`: Windows related. | ||||
|  | ||||
| ### Area | ||||
|  | ||||
| * `area/acme`: ACME related. | ||||
| * `area/api`: Traefik API related. | ||||
| * `area/authentication`: Authentication related. | ||||
| * `area/cluster`: Traefik clustering related. | ||||
| * `area/documentation`: Documentation related. | ||||
| * `area/infrastructure`: CI or Traefik building scripts related. | ||||
| * `area/healthcheck`: Health-check related. | ||||
| * `area/logs`: Logs related. | ||||
| * `area/middleware`: Middleware related. | ||||
| * `area/middleware/metrics`: Metrics related. (Prometheus, StatsD, ...) | ||||
| * `area/middleware/tracing`: Tracing related. (Jaeger, Zipkin, ...) | ||||
| * `area/oxy`: Oxy related. | ||||
| * `area/provider`: related to all providers. | ||||
| * `area/provider/boltdb`: Boltd DB related. | ||||
| * `area/provider/consul`: Consul related. | ||||
| * `area/provider/docker`: Docker and Swarm related. | ||||
| * `area/provider/ecs`: ECS related. | ||||
| * `area/provider/etcd`: Etcd related. | ||||
| * `area/provider/eureka`: Eureka related. | ||||
| * `area/provider/file`: file provider related. | ||||
| * `area/provider/k8s`: Kubernetes related. | ||||
| * `area/provider/kv`: KV related. | ||||
| * `area/provider/marathon`: Marathon related. | ||||
| * `area/provider/mesos`: Mesos related. | ||||
| * `area/provider/rancher`: Rancher related. | ||||
| * `area/provider/servicefabric`: Azure service fabric related. | ||||
| * `area/provider/zk`: Zoo Keeper related. | ||||
| * `area/rules`: Rules related. | ||||
| * `area/server`: Server related. | ||||
| * `area/sticky-session`: Sticky session related. | ||||
| * `area/tls`: TLS related. | ||||
| * `area/websocket`: WebSocket related. | ||||
| * `area/webui`: Web UI related. | ||||
|  | ||||
| ### Issues Priority | ||||
|  | ||||
| * `priority/P0`: needs hot fix. | ||||
| * `priority/P1`: need to be fixed in next release. | ||||
| * `priority/P2`: need to be fixed in the future. | ||||
| * `priority/P3`: maybe. | ||||
|  | ||||
| ### PR size | ||||
|  | ||||
| Automatically set by a bot. | ||||
|  | ||||
| * `size/S`: small PR. | ||||
| * `size/M`: medium PR. | ||||
| * `size/L`: Large PR. | ||||
| Please read the [maintainer's guidelines](maintainers-guidelines.md). | ||||
|   | ||||
| @@ -1,44 +1,63 @@ | ||||
| --- | ||||
| title: "Traefik Submitting Issues Documentation" | ||||
| description: "Help us help you! Learn how to submit an issue, following the guidelines, so the Traefik Proxy team can help. Read the technical documentation." | ||||
| --- | ||||
|  | ||||
| # Submitting Issues | ||||
|  | ||||
| Help Us Help You! | ||||
| {: .subtitle } | ||||
|  | ||||
| We use the [GitHub issue tracker](https://github.com/containous/traefik/issues) to keep track of issues in Traefik.  | ||||
| Issues are perfect for requesting a feature/enhancement or reporting a suspected bug. | ||||
| We use the [GitHub issue tracker](https://github.com/traefik/traefik/issues) to keep track of issues in Traefik. | ||||
|  | ||||
| The process of sorting and checking the issues is a daunting task, and requires a lot of work (more than an hour a day ... just for sorting). | ||||
| To save us some time and get quicker feedback, be sure to follow the guide lines below. | ||||
| The process of sorting and checking the issues is a daunting task, and requires a lot of work. | ||||
| To help maintainers (and other community members) quickly and effortlessly understand what you need, | ||||
| be sure to follow the guidelines below. | ||||
|  | ||||
| !!! important "Getting Help Vs Reporting an Issue" | ||||
|  | ||||
|     The issue tracker is not a general support forum, but a place to report bugs and asks for new features. | ||||
|  | ||||
|     For end-user related support questions, try using first: | ||||
|  | ||||
|     - the Traefik community forum: [](https://community.containo.us/) | ||||
|     For end-user related support questions, try using the [Traefik Community Forum](https://community.traefik.io/) | ||||
|    [](https://community.traefik.io/) | ||||
|  | ||||
| ## Issue Title | ||||
|  | ||||
| The title must be short and descriptive. (~60 characters) | ||||
|  | ||||
| ## Description | ||||
| Examples: | ||||
|  | ||||
| Follow the [issue template](https://github.com/containous/traefik/blob/master/.github/ISSUE_TEMPLATE.md) as much as possible. | ||||
|  | ||||
| Explain us in which conditions you encountered the issue, what is your context. | ||||
|  | ||||
| Remain as clear and concise as possible | ||||
|  | ||||
| Take time to polish the format of your message so we'll enjoy reading it and working on it.  | ||||
| Help the readers focus on what matters, and help them understand the structure of your message (see the [Github Markdown Syntax](https://help.github.com/articles/github-flavored-markdown)). | ||||
| * Bug: Duplicate requests in access logs | ||||
| * Feature: Support TCP | ||||
|  | ||||
| ## Feature Request | ||||
|  | ||||
| Traefik is an open-source project and aims to be the best edge router possible. | ||||
| Traefik is an open source project and aims to be the best edge router possible. | ||||
|  | ||||
| Remember when asking for new features that these must be useful to the majority (and not only useful in edge case scenarios, or hack-like setups). | ||||
| Follow the [issue template](https://github.com/traefik/traefik/blob/master/.github/ISSUE_TEMPLATE/feature-request.yml) as much as possible. | ||||
|  | ||||
| Do you best to explain what you're looking for, and why it would improve Traefik for everyone.  | ||||
| Do your best to explain what you're looking for, and why it would improve Traefik for everyone. | ||||
| Be detailed and share the use-case(s) to allow us to see the value of your feature request as quickly as possible. | ||||
|  | ||||
| Features with a lot of positive interaction (claps, +1s, conversation about how this would impact them) indicate higher community interest and help us to prioritize. | ||||
|  | ||||
| If you are interested in creating a PR for your feature request, let us know in the issue, so we can work with you. | ||||
| It can take a lot of work to make sure a PR can integrate with our existing code and planning with the team ahead of time can make sure that your PR can be accepted and merged quickly. | ||||
|  | ||||
| ## Issues or Possible Bug Reports | ||||
|  | ||||
| Follow the [issue template](https://github.com/traefik/traefik/blob/master/.github/ISSUE_TEMPLATE/bug_report.yml) as much as possible. | ||||
|  | ||||
| Explain the conditions in which you encountered the issue; what is your context? | ||||
| Share any logs you may have, and make sure to share the steps it takes to reproduce your issue or bug. | ||||
|  | ||||
| Remain as clear and concise as possible. | ||||
|  | ||||
| Take time to polish the format of your message, so we'll enjoy reading it and working on it. | ||||
| Help your readers focus on what matters and help them understand the structure of your message (see the [GitHub Markdown Syntax](https://docs.github.com/en/get-started/writing-on-github)). | ||||
|  | ||||
| ## International English | ||||
|  | ||||
| Every maintainer / Traefik user is not a native English speaker, so if you feel sometimes that some messages sound rude, remember that it probably is a language barrier problem from someone willing to help you. | ||||
| Every maintainer / Traefik user is not a native English speaker, so if you sometimes feel that some messages sound rude, remember that it probably is a language barrier problem from someone willing to help you. | ||||
|   | ||||
| @@ -1,45 +1,228 @@ | ||||
| # Submitting Pull Requests | ||||
| --- | ||||
| title: "Traefik Pull Requests Documentation" | ||||
| description: "Looking to contribute to Traefik Proxy? This guide will show you the guidelines for submitting a PR in our contributors guide repository." | ||||
| --- | ||||
|  | ||||
| A Quick Guide for Efficient Contributions | ||||
| {: .subtitle } | ||||
| # Before You Submit a Pull Request | ||||
|  | ||||
| So you've decided to improve Traefik?  | ||||
| Thank You!  | ||||
| Now the last step is to submit your Pull Request in a way that makes sure it gets the attention it deserves. | ||||
| This guide is for contributors who already have a pull request to submit. | ||||
| If you are looking for information on setting up your developer environment | ||||
| and creating code to contribute to Traefik Proxy or related projects, | ||||
| see the [development guide](https://docs.traefik.io/contributing/building-testing/). | ||||
|  | ||||
| Let's go through the classic pitfalls to make sure everything is right.  | ||||
| Looking for a way to contribute to Traefik Proxy? | ||||
| Check out this list of [Priority Issues](https://github.com/traefik/traefik/labels/contributor%2Fwanted), | ||||
| the [Good First Issue](https://github.com/traefik/traefik/labels/contributor%2Fgood-first-issue) list, | ||||
| or the list of [confirmed bugs](https://github.com/traefik/traefik/labels/kind%2Fbug%2Fconfirmed) waiting to be remedied. | ||||
|  | ||||
| ## Title | ||||
| ## How We Prioritize | ||||
|  | ||||
| The title must be short and descriptive. (~60 characters) | ||||
| We wish we could review every pull request right away, but because it's a time consuming operation, it's not always possible. | ||||
|  | ||||
| ## Description | ||||
| The PRs we are able to handle the fastest are: | ||||
|  | ||||
| Follow the [pull request template](https://github.com/containous/traefik/blob/master/.github/PULL_REQUEST_TEMPLATE.md) as much as possible. | ||||
| * Documentation updates. | ||||
| * Bug fixes. | ||||
| * Enhancements and Features with a `contributor/wanted` tag. | ||||
|  | ||||
| Explain the conditions which led you to write this PR: give us context. | ||||
| The context should lead to something, an idea or a problem that you’re facing. | ||||
| PRs that take more time to address include: | ||||
|  | ||||
| Remain clear and concise. | ||||
| * Enhancements or Features without the `contributor/wanted` tag. | ||||
|  | ||||
| Take time to polish the format of your message so we'll enjoy reading it and working on it. | ||||
| Help the readers focus on what matters, and help them understand the structure of your message (see the [Github Markdown Syntax](https://help.github.com/articles/github-flavored-markdown)). | ||||
| If you have an idea for an enhancement or feature that you would like to build, | ||||
| [create an issue](https://github.com/traefik/traefik/issues/new/choose) for it first | ||||
| and tell us you are interested in writing the PR. | ||||
| If an issue already exists, definitely comment on it to tell us you are interested in creating a PR. | ||||
|  | ||||
| ## PR Content | ||||
| This will allow us to communicate directly and let you know if it is something we would accept. | ||||
|  | ||||
| - Make it small. | ||||
| - One feature per Pull Request. | ||||
| - Write useful descriptions and titles. | ||||
| - Avoid re-formatting code that is not on the path of your PR. | ||||
| - Make sure the [code builds](building-testing.md). | ||||
| - Make sure [all tests pass](building-testing.md). | ||||
| - Add tests. | ||||
| - Address review comments in terms of additional commits (and don't amend/squash existing ones unless the PR is trivial). | ||||
| It also allows us to make sure you have all the information you need during the design phase | ||||
| so that it can be reviewed and merged quickly. | ||||
|  | ||||
| !!! note "Third-Party Dependencies" | ||||
| Read more about the [Triage process](https://github.com/traefik/contributors-guide/blob/master/issue_triage.md) in the docs. | ||||
|  | ||||
|     If a PR involves changes to third-party dependencies, the commits pertaining to the vendor folder and the manifest/lock file(s) should be committed separated. | ||||
| ## The Pull Request Submit Process | ||||
|  | ||||
| !!! tip "10 Tips for Better Pull Requests" | ||||
| Merging a PR requires the following steps to be completed before it is merged automatically. | ||||
|  | ||||
|     We enjoyed this article, maybe you will too! [10 tips for better pull requests](https://blog.ploeh.dk/2015/01/15/10-tips-for-better-pull-requests/). | ||||
| * Make sure your pull request adheres to our best practices. These include: | ||||
|     * [Following project conventions](https://github.com/traefik/traefik/blob/master/docs/content/contributing/maintainers-guidelines.md); including using the PR Template. | ||||
|     * Make small pull requests. | ||||
|     * Solve only one problem at a time. | ||||
|     * Comment thoroughly. | ||||
|     * Do not open the PR from an organization repository. | ||||
|     * Keep "allows edit from maintainer" checked. | ||||
|     * Use semantic line breaks for documentation. | ||||
|     * Ensure your PR is not a draft. We do not review drafts, but do answer questions and confer with developers on them as needed. | ||||
| * Pass the validation check. | ||||
| * Pass all tests. | ||||
| * Receive 3 approving reviews from maintainers. | ||||
|  | ||||
| ## Pull Request Review Cycle | ||||
|  | ||||
| Learn about our [Triage Process](https://github.com/traefik/contributors-guide/blob/master/issue_triage.md), | ||||
| in short, it looks like this: | ||||
|  | ||||
| * We triage every new PR or comment before entering it into the review process. | ||||
|     * We ensure that all prerequisites for review have been met. | ||||
|     * We check to make sure the use case meets our needs. | ||||
|     * We assign reviewers. | ||||
| * Design Review. | ||||
|     * This takes longer than other parts of the process. | ||||
|     * We review that there are no obvious conflicts with our codebase. | ||||
| * Code Review. | ||||
|     * We review the code in-depth and run tests. | ||||
|     * We may ask for changes here. | ||||
|     * During code review, we ask that you be reasonably responsive, | ||||
|       if a PR languishes in code review it is at risk of rejection, | ||||
|       or we may take ownership of the PR and the contributor will become a co-author. | ||||
| * Merge. | ||||
|     * Success! | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     Occasionally, we may freeze our codebase when working towards a specific feature or goal that could impact other development. | ||||
|     During this time, your pull request could remain unmerged while the release work is completed. | ||||
|  | ||||
| ## Run Local Verifications | ||||
|  | ||||
| You must run these local verifications before you submit your pull request to predict the pass or failure of continuous integration. | ||||
| Your PR will not be reviewed until these are green on the CI. | ||||
|  | ||||
| * `make validate` | ||||
| * `make pull-images` | ||||
| * `make test` | ||||
|  | ||||
| ## The Testing and Merge Workflow | ||||
|  | ||||
| Pull Requests are managed by the bot [Myrmica Lobicornis](https://github.com/traefik/lobicornis). | ||||
| This bot is responsible for verifying GitHub Checks (CI, Tests, etc), mergability, and minimum reviews. | ||||
| In addition, it rebases or merges with the base PR branch if needed. | ||||
| It performs several other housekeeping items | ||||
| and you can read more about those on the [README](https://github.com/traefik/lobicornis) for Lobicornis. | ||||
|  | ||||
| The maintainer giving the final LGTM must add the `status/3-needs-merge` label to trigger the merge bot. | ||||
|  | ||||
| By default, a squash-rebase merge will be carried out. | ||||
|  | ||||
| The status `status/4-merge-in-progress` is only used by the bot. | ||||
|  | ||||
| If the bot is not able to perform the merge, the label `bot/need-human-merge` is added. | ||||
| In such a situation, solve the conflicts/CI/... and then remove the label `bot/need-human-merge`. | ||||
|  | ||||
| To prevent the bot from automatically merging a PR, add the label `bot/no-merge`. | ||||
|  | ||||
| The label `bot/light-review` decreases the number of required LGTM from 3 to 1. | ||||
|  | ||||
| This label can be used when: | ||||
|  | ||||
| * Updating a dependency. | ||||
| * Merging branches back into the next version branch. | ||||
| * Submitting minor documentation changes. | ||||
| * Submitting changelog PRs. | ||||
|  | ||||
| ## Why Was My Pull Request Closed? | ||||
|  | ||||
| Traefik Proxy is made by the community for the community, | ||||
| as such the goal is to engage the community to make Traefik the best reverse proxy available. | ||||
| Part of this goal is maintaining a lean codebase and ensuring code velocity. | ||||
| unfortunately, this means that sometimes we will not be able to merge a pull request. | ||||
|  | ||||
| Because we respect the work you did, you will always be told why we are closing your pull request. | ||||
| If you do not agree with our decision, do not worry; closed pull requests are effortless to recreate, | ||||
| and little work is lost by closing a pull request that subsequently needs to be reopened. | ||||
|  | ||||
| Your pull request might be closed if: | ||||
|  | ||||
| * Your PR's design conflicts with our existing codebase in such a way that merging is not an option | ||||
|   and the work needed to make your pull request usable is too high. | ||||
|     * To prevent this, make sure you created an issue first | ||||
|       and think about including Traefik Proxy maintainers in your design phase to minimize conflicts. | ||||
| * Your PR is for an enhancement or feature that we will not use. | ||||
|     * Please remember to create an issue for any pull request **before** you create a PR | ||||
|       to ensure that your goal is something we can merge and that you have any design insight you might need from the team. | ||||
| * Your PR has been waiting for feedback from the contributor for over 90 days. | ||||
|  | ||||
| ## Why is My Pull Request Not Getting Reviewed | ||||
|  | ||||
| A few factors affect how long your pull request might wait for review. | ||||
|  | ||||
| We must prioritize which PRs we focus on. | ||||
| Our first priority is PRs we have identified as having high community engagement and broad applicability. | ||||
| We put our top priorities on our roadmap, and you can identify them by the `contributor/wanted` tag. | ||||
| These PRs will enter our review process the fastest. | ||||
|  | ||||
| Our second priority is bug fixes. | ||||
| Especially for bugs that have already been tagged with `bug/confirmed`. | ||||
| These reviews enter the process quickly. | ||||
|  | ||||
| If your PR does not meet the criteria above, | ||||
| it will take longer for us to review, as any PRs that do meet the criteria above will be prioritized. | ||||
|  | ||||
| Additionally, during the last few weeks of a milestone, we stop reviewing PRs to reduce churn and stabilize. | ||||
| We will resume after the release. | ||||
|  | ||||
| The second major reason that we deprioritize your PR is that you are not following best practices. | ||||
|  | ||||
| The most common failures to follow best practices are: | ||||
|  | ||||
| * You did not create an issue for the PR you wish to make. | ||||
|   If you do not create an issue before submitting your PR, | ||||
|   we will not be able to answer any design questions and let you know how likely your PR is to be merged. | ||||
| * You created pull requests that are too large to review. | ||||
|     * Break your pull requests up. | ||||
|       If you can extract whole ideas from your pull request and send those as pull requests of their own, | ||||
|       you should do that instead. | ||||
|       It is better to have many pull requests addressing one thing than one pull request addressing many things. | ||||
|     * Traefik Proxy is a fast-moving codebase — lock in your changes ASAP with your small pull request, | ||||
|       and make merges be someone else's problem. | ||||
|       We want every pull request to be useful on its own, | ||||
|       so use your best judgment on what should be a pull request vs. a commit. | ||||
| * You did not comment well. | ||||
|     * Comment everything. | ||||
|  | ||||
| Please remember that we are working internationally, cross-culturally, and with different use-cases. | ||||
| Your reviewer will not intuitively understand the problem the same way you do or solve it the same way you would. | ||||
| This is why every change you make must be explained, and your strategy for coding must also be explained. | ||||
|  | ||||
| * Your tests were inadequate or absent. | ||||
|     * If you do not know how to test your PR, please ask! | ||||
|       We will be happy to help you or suggest appropriate test cases. | ||||
|  | ||||
| If you have already followed the best practices and your PR still has not received a response, | ||||
| here are some things you can do to move the process along: | ||||
|  | ||||
| * If you have fixed all the issues from a review, | ||||
|   remember to re-request a review (using the designated button) to let your reviewer know that you are ready. | ||||
|   You can choose to comment with the changes you made. | ||||
| * Kindly comment on the pull request. Doing so will automatically give your PR visibility during the triage process. | ||||
|  | ||||
| For more information on best practices, try these links: | ||||
|  | ||||
| * [How to Write a Git Commit Message - Chris Beams](https://chris.beams.io/posts/git-commit/) | ||||
| * [Distributed Git - Contributing to a Project (Commit Guidelines)](https://git-scm.com/book/en/v2/Distributed-Git-Contributing-to-a-Project) | ||||
| * [What’s with the 50/72 rule? - Preslav Rachev](https://preslav.me/2015/02/21/what-s-with-the-50-72-rule/) | ||||
| * [A Note About Git Commit Messages - Tim Pope](https://tbaggery.com/2008/04/19/a-note-about-git-commit-messages.html) | ||||
|  | ||||
| ## It's OK to Push Back | ||||
|  | ||||
| Sometimes reviewers make mistakes. | ||||
| It is OK to push back on changes your reviewer requested. | ||||
| If you have a good reason for doing something a certain way, you are absolutely allowed to debate the merits of a requested change. | ||||
| Both the reviewer and reviewee should strive to discuss these issues in a polite and respectful manner. | ||||
|  | ||||
| You might be overruled, but you might also prevail. | ||||
| We are pretty reasonable people. | ||||
|  | ||||
| Another phenomenon of open-source projects (where anyone can comment on any issue) is the dog-pile - | ||||
| your pull request gets so many comments from so many people it becomes hard to follow. | ||||
| In this situation, you can ask the primary reviewer (assignee) whether they want you to fork a new pull request | ||||
| to clear out all the comments. | ||||
| You do not have to fix every issue raised by every person who feels like commenting, | ||||
| but you should answer reasonable comments with an explanation. | ||||
|  | ||||
| ## Common Sense and Courtesy | ||||
|  | ||||
| No document can take the place of common sense and good taste. | ||||
| Use your best judgment, while you put a bit of thought into how your work can be made easier to review. | ||||
| If you do these things, your pull requests will get merged with less friction. | ||||
|   | ||||
| @@ -1,3 +1,8 @@ | ||||
| --- | ||||
| title: "Traefik Security Documentation" | ||||
| description: "Security is a key part of Traefik Proxy. Read the technical documentation to learn about security advisories, CVE, and how to report a vulnerability." | ||||
| --- | ||||
|  | ||||
| # Security | ||||
|  | ||||
| ## Security Advisories | ||||
| @@ -7,10 +12,12 @@ You can subscribe sending a mail to security+subscribe@traefik.io or on [the onl | ||||
|  | ||||
| ## CVE | ||||
|  | ||||
| Reported vulnerabilities can be found on  | ||||
| Reported vulnerabilities can be found on | ||||
| [cve.mitre.org](https://cve.mitre.org/cgi-bin/cvekey.cgi?keyword=traefik). | ||||
|  | ||||
| ## Report a Vulnerability | ||||
|  | ||||
| We want to keep Traefik safe for everyone. | ||||
| If you've discovered a security vulnerability in Traefik, we appreciate your help in disclosing it to us in a responsible manner, using [this form](https://security.traefik.io). | ||||
| If you've discovered a security vulnerability in Traefik, | ||||
| we appreciate your help in disclosing it to us in a responsible manner, | ||||
| by creating a [security advisory](https://github.com/traefik/traefik/security/advisories). | ||||
|   | ||||
| @@ -1,27 +1,32 @@ | ||||
| --- | ||||
| title: "Traefik Contribution Documentation" | ||||
| description: "Thank you to all those who have contributed! Traefik Proxy is an open-source project that thrives with the support of our passionate community." | ||||
| --- | ||||
|  | ||||
| # Thank You! | ||||
|  | ||||
| _You_ Made It | ||||
| {: .subtitle} | ||||
|  | ||||
| Traefik truly is an [open-source project](https://github.com/containous/traefik/), | ||||
| and wouldn't have become what it is today without the help of our [many contributors](https://github.com/containous/traefik/graphs/contributors) (at the time of writing this), | ||||
| not accounting for people having helped with issues, tests, comments, articles, ... or just enjoying it and letting others know. | ||||
| Traefik Proxy truly is an [open-source project](https://github.com/traefik/traefik/), | ||||
| and wouldn't have become what it is today without the help of our [many contributors](https://github.com/traefik/traefik/graphs/contributors), | ||||
| not accounting for people having helped with issues, tests, comments, articles, ... or just enjoy using Traefik Proxy and share with others. | ||||
|  | ||||
| So once again, thank you for your invaluable help on making Traefik such a good product. | ||||
| So once again, thank you for your invaluable help in making Traefik such a good product! | ||||
|  | ||||
| !!! question "Where to Go Next?" | ||||
|     If you want to: | ||||
|  | ||||
|     - Propose and idea, request a feature a report a bug, | ||||
|       read the page [Submitting Issues](./submitting-issues.md). | ||||
|     - Propose an idea, request a feature, or report a bug,  | ||||
|       then read [Submitting Issues](./submitting-issues.md). | ||||
|     - Discover how to make an efficient contribution, | ||||
|       read the page [Submitting Pull Requests](./submitting-pull-requests.md). | ||||
|       then read [Submitting Pull Requests](./submitting-pull-requests.md). | ||||
|     - Learn how to build and test Traefik, | ||||
|       the page [Building and Testing](./building-testing.md) is for you. | ||||
|       then the page [Building and Testing](./building-testing.md) is for you. | ||||
|     - Contribute to the documentation, | ||||
|       read the related page [Documentation](./documentation.md). | ||||
|       then read the page about [Documentation](./documentation.md). | ||||
|     - Understand how do we learn about Traefik usage, | ||||
|       read the [Data Collection](./data-collection.md) page. | ||||
|     - Spread the love about Traefik, please check the [Advocating](./advocating.md) page. | ||||
|     - Learn about who are the maintainers and how they work on the project, | ||||
|       read the [Maintainers](./maintainers.md) page. | ||||
|       read the [Maintainers](./maintainers.md) and [Maintainer Guidelines](./maintainers-guidelines.md) pages. | ||||
|   | ||||
							
								
								
									
										44
									
								
								docs/content/deprecation/features.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								docs/content/deprecation/features.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| # Feature Deprecation Notices | ||||
|  | ||||
| This page is maintained and updated periodically to reflect our roadmap and any decisions around feature deprecation. | ||||
|  | ||||
| | Feature                                                                                                     | Deprecated | End of Support | Removal | | ||||
| |-------------------------------------------------------------------------------------------------------------|------------|----------------|---------| | ||||
| | [Pilot](#pilot)                                                                                             | 2.7        | 2.8            | 2.9     | | ||||
| | [Consul Enterprise Namespace](#consul-enterprise-namespace)                                                 | 2.8        | N/A            | 3.0     | | ||||
| | [TLS 1.0 and 1.1 Support](#tls-10-and-11)                                                                   | N/A        | 2.8            | N/A     | | ||||
| | [Nomad Namespace](#nomad-namespace)                                                                         | 2.10       | N/A            | 3.0     | | ||||
| | [Kubernetes CRDs API Group `traefik.containo.us`](#kubernetes-crd-provider-api-group-traefikcontainous)     | 2.10       | N/A            | 3.0     | | ||||
| | [Kubernetes CRDs API Version `traefik.io/v1alpha1`](#kubernetes-crd-provider-api-version-traefikiov1alpha1) | 3.0        | N/A            | 4.0     | | ||||
|  | ||||
| ## Impact | ||||
|  | ||||
| ### Pilot | ||||
|  | ||||
| Metrics will continue to function normally up to 2.8, when they will be disabled.   | ||||
| In 2.9, the Pilot platform and all Traefik integration code will be permanently removed. | ||||
|  | ||||
| Starting on 2.7 the pilot token will not be a requirement anymore for plugins.   | ||||
| Since 2.8, a [new plugin catalog](https://plugins.traefik.io) is available, decoupled from Pilot. | ||||
|  | ||||
| ### Consul Enterprise Namespace | ||||
|  | ||||
| Starting on 2.8 the `namespace` option of Consul and Consul Catalog providers is deprecated,  | ||||
| please use the `namespaces` options instead.   | ||||
|  | ||||
| ### TLS 1.0 and 1.1 | ||||
|  | ||||
| Starting on 2.8 the default TLS options will use the minimum version of TLS 1.2. Of course, it can still be overridden with custom configuration.   | ||||
|  | ||||
| ### Nomad Namespace | ||||
|  | ||||
| Starting on 2.10 the `namespace` option of the Nomad provider is deprecated, | ||||
| please use the `namespaces` options instead. | ||||
|  | ||||
| ### Kubernetes CRD Provider API Group `traefik.containo.us` | ||||
|  | ||||
| In v2.10, the Kubernetes CRD provider API Group `traefik.containo.us` is deprecated, and its support will end starting with Traefik v3. Please use the API Group `traefik.io` instead. | ||||
|  | ||||
| ### Kubernetes CRD Provider API Version `traefik.io/v1alpha1` | ||||
|  | ||||
| The Kubernetes CRD provider API Version `traefik.io/v1alpha1` will subsequently be deprecated in Traefik v3. The next version will be `traefik.io/v1`. | ||||
							
								
								
									
										41
									
								
								docs/content/deprecation/releases.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								docs/content/deprecation/releases.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| # Releases | ||||
|  | ||||
| ## Versions | ||||
|  | ||||
| Below is a non-exhaustive list of versions and their maintenance status: | ||||
|  | ||||
| | Version | Release Date | Active Support     | Security Support |  | ||||
| |---------|--------------|--------------------|------------------| | ||||
| | 2.10    | Apr 24, 2023 | Yes                | Yes              | | ||||
| | 2.9     | Oct 03, 2022 | Ended Apr 24, 2023 | No               | | ||||
| | 2.8     | Jun 29, 2022 | Ended Oct 03, 2022 | No               | | ||||
| | 2.7     | May 24, 2022 | Ended Jun 29, 2022 | No               | | ||||
| | 2.6     | Jan 24, 2022 | Ended May 24, 2022 | No               | | ||||
| | 2.5     | Aug 17, 2021 | Ended Jan 24, 2022 | No               | | ||||
| | 2.4     | Jan 19, 2021 | Ended Aug 17, 2021 | No               | | ||||
| | 2.3     | Sep 23, 2020 | Ended Jan 19, 2021 | No               | | ||||
| | 2.2     | Mar 25, 2020 | Ended Sep 23, 2020 | No               | | ||||
| | 2.1     | Dec 11, 2019 | Ended Mar 25, 2020 | No               | | ||||
| | 2.0     | Sep 16, 2019 | Ended Dec 11, 2019 | No               | | ||||
| | 1.7     | Sep 24, 2018 | Ended Dec 31, 2021 | Contact Support  | | ||||
|  | ||||
| ??? example "Active Support / Security Support" | ||||
|  | ||||
|     **Active support**: receives any bug fixes. | ||||
|     **Security support**: receives only critical bug and security fixes. | ||||
|  | ||||
| This page is maintained and updated periodically to reflect our roadmap and any decisions affecting the end of support for Traefik Proxy. | ||||
|  | ||||
| Please refer to our migration guides for specific instructions on upgrading between versions, an example is the [v1 to v2 migration guide](../migration/v1-to-v2.md). | ||||
|  | ||||
| !!! important "All target dates for end of support or feature removal announcements may be subject to change." | ||||
|  | ||||
| ## Versioning Scheme | ||||
|  | ||||
| The Traefik Proxy project follows the [semantic versioning](https://semver.org/) scheme and maintains a separate branch for each minor version. The main branch always represents the next upcoming minor or major version. | ||||
|  | ||||
| And these are our guiding rules for version support: | ||||
|  | ||||
| - **Only the latest `minor`** will be on active support at any given time | ||||
| - **The last `minor` after releasing a new `major`** will be supported for 1 year following the `major` release | ||||
| - **Previous rules are subject to change** and in such cases an announcement will be made publicly, [here](https://traefik.io/blog/traefik-2-1-in-the-wild/) is an example extending v1.x branch support. | ||||
| @@ -1,14 +1,34 @@ | ||||
| --- | ||||
| title: Concepts | ||||
| description: Traefik - base concepts and main features | ||||
| --- | ||||
|  | ||||
| # Concepts | ||||
|  | ||||
| Everything You Need to Know | ||||
| {: .subtitle } | ||||
| This page explains the base concepts of Traefik. | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Introduction | ||||
|  | ||||
| Traefik is based on the concept of EntryPoints, Routers, Middlewares and Services. | ||||
|  | ||||
| The main features include dynamic configuration, automatic service discovery, and support for multiple backends and protocols. | ||||
|  | ||||
| 1. [EntryPoints](../routing/entrypoints.md "Link to docs about EntryPoints"): EntryPoints are the network entry points into Traefik. They define the port which will receive the packets, and whether to listen for TCP or UDP. | ||||
|  | ||||
| 2. [Routers](../routing/routers/index.md "Link to docs about routers"): A router is in charge of connecting incoming requests to the services that can handle them. | ||||
|  | ||||
| 3. [Middlewares](../middlewares/overview.md "Link to docs about middlewares"): Attached to the routers, middlewares can modify the requests or responses before they are sent to your service | ||||
|  | ||||
| 4. [Services](../routing/services/index.md "Link to docs about services"): Services are responsible for configuring how to reach the actual services that will eventually handle the incoming requests. | ||||
|  | ||||
| ## Edge Router | ||||
|  | ||||
| Traefik is an _Edge Router_, it means that it's the door to your platform, and that it intercepts and routes every incoming request: | ||||
| it knows all the logic and every rule that determine which services handle which requests (based on the [path](../routing/routers/index.md#rule), the [host](../routing/routers/index.md#rule), [headers](../routing/routers/index.md#rule), [and so on](../routing/routers/index.md#rule) ...). | ||||
| Traefik is an *Edge Router*, it means that it's the door to your platform, and that it intercepts and routes every incoming request: | ||||
| it knows all the logic and every [rule](../routing/routers/index.md#rule "Link to docs about routing rules") that determine which services handle which requests (based on the *path*, the *host*, *headers*, etc.). | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Auto Service Discovery | ||||
|  | ||||
| @@ -16,21 +36,25 @@ Where traditionally edge routers (or reverse proxies) need a configuration file | ||||
|  | ||||
| Deploying your services, you attach information that tells Traefik the characteristics of the requests the services can handle. | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| It means that when a service is deployed, Traefik detects it immediately and updates the routing rules in real time. | ||||
| The opposite is true: when you remove a service from your infrastructure, the route will disappear accordingly. | ||||
| Similarly, when a service is removed from the infrastructure, the corresponding route is deleted accordingly. | ||||
|  | ||||
| You no longer need to create and synchronize configuration files cluttered with IP addresses or other rules. | ||||
|  | ||||
| !!! info "Many different rules" | ||||
|  | ||||
|     In the example above, we used the request [path](../routing/routers/index.md#rule) to determine which service was in charge, but of course you can use many other different [rules](../routing/routers/index.md#rule). | ||||
|     In the example above, we used the request [path rule](../routing/routers/index.md#rule "Link to docs about routing rules") to determine which service was in charge. | ||||
|     Certainly, you can use many other different [rules](../routing/routers/index.md#rule "Link to docs about routing rules"). | ||||
|  | ||||
| !!! info "Updating the requests" | ||||
|  | ||||
|     In the [middleware](../middlewares/overview.md) section, you can learn about how to update the requests before forwarding them to the services. | ||||
|     In the [middleware](../middlewares/overview.md "Link to middleware documentation") section, you can learn about how to update the requests before forwarding them to the services. | ||||
|  | ||||
| !!! question "How does Traefik discover the services?" | ||||
|  | ||||
|     Traefik is able to use your cluster API to discover the services and read the attached information. In Traefik, these connectors are called [providers](../providers/overview.md) because they _provide_ the configuration to Traefik. To learn more about them, read the [provider overview](../providers/overview.md) section. | ||||
|     Traefik is able to use your cluster API to discover the services and read the attached information. | ||||
|     In Traefik, these connectors are called [providers](../providers/overview.md "Link to overview about Traefik providers") because they *provide* the configuration to Traefik. | ||||
|  | ||||
| {!traefik-for-business-applications.md!} | ||||
|   | ||||
| @@ -1,3 +1,8 @@ | ||||
| --- | ||||
| title: "Traefik Configuration Documentation" | ||||
| description: "Get started with Traefik Proxy. This page will introduce you to the dynamic routing and startup configurations. Read the technical documentation." | ||||
| --- | ||||
|  | ||||
| # Configuration Introduction | ||||
|  | ||||
| How the Magic Happens | ||||
| @@ -13,13 +18,13 @@ Configuration in Traefik can refer to two different things: | ||||
| Elements in the _static configuration_ set up connections to [providers](../providers/overview.md) and define the [entrypoints](../routing/entrypoints.md) Traefik will listen to (these elements don't change often). | ||||
|  | ||||
| The _dynamic configuration_ contains everything that defines how the requests are handled by your system. | ||||
| This configuration can change and is seamlessly hot-reloaded, without any request interruption or connection loss.     | ||||
| This configuration can change and is seamlessly hot-reloaded, without any request interruption or connection loss. | ||||
|  | ||||
| !!! warning "Incompatible Configuration" | ||||
|     Please be aware that the old configurations for Traefik v1.x are NOT compatible with the v2.x config as of now. | ||||
|     If you are running v2, please ensure you are using a v2 configuration. | ||||
|  | ||||
| ## The Dynamic Configuration  | ||||
| ## The Dynamic Configuration | ||||
|  | ||||
| Traefik gets its _dynamic configuration_ from [providers](../providers/overview.md): whether an orchestrator, a service registry, or a plain old configuration file. | ||||
|  | ||||
| @@ -28,14 +33,14 @@ Since this configuration is specific to your infrastructure choices, we invite y | ||||
| !!! info "" | ||||
|  | ||||
|     In the [Quick Start example](../getting-started/quick-start.md), the dynamic configuration comes from docker in the form of labels attached to your containers. | ||||
|      | ||||
|  | ||||
| !!! info "HTTPS Certificates also belong to the dynamic configuration." | ||||
|      | ||||
|     You can add / update / remove them without restarting your Traefik instance.  | ||||
|   | ||||
|  | ||||
|     You can add / update / remove them without restarting your Traefik instance. | ||||
|  | ||||
| ## The Static Configuration | ||||
|  | ||||
| There are three different, **mutually exclusive** (e.g. you can use only one at the same time), ways to define static configuration options in Traefik: | ||||
| There are three different, **mutually exclusive** (i.e. you can use only one at the same time), ways to define static configuration options in Traefik: | ||||
|  | ||||
| 1. In a configuration file | ||||
| 1. In the command-line arguments | ||||
| @@ -45,13 +50,13 @@ These ways are evaluated in the order listed above. | ||||
|  | ||||
| If no value was provided for a given option, a default value applies. | ||||
| Moreover, if an option has sub-options, and any of these sub-options is not specified, a default value will apply as well. | ||||
|      | ||||
|  | ||||
| For example, the `--providers.docker` option is enough by itself to enable the docker provider, even though sub-options like `--providers.docker.endpoint` exist. | ||||
| Once positioned, this option sets (and resets) all the default values of the sub-options of `--providers.docker`. | ||||
|      | ||||
|  | ||||
| ### Configuration File | ||||
|  | ||||
| At startup, Traefik searches for a file named `traefik.toml` (or `traefik.yml` or `traefik.yaml`) in: | ||||
| At startup, Traefik searches for static configuration in a file named `traefik.yml` (or `traefik.yaml` or `traefik.toml`) in: | ||||
|  | ||||
| - `/etc/traefik/` | ||||
| - `$XDG_CONFIG_HOME/` | ||||
| @@ -61,7 +66,7 @@ At startup, Traefik searches for a file named `traefik.toml` (or `traefik.yml` o | ||||
| You can override this using the `configFile` argument. | ||||
|  | ||||
| ```bash | ||||
| traefik --configFile=foo/bar/myconfigfile.toml | ||||
| traefik --configFile=foo/bar/myconfigfile.yml | ||||
| ``` | ||||
|  | ||||
| ### Arguments | ||||
| @@ -74,17 +79,19 @@ traefik --help | ||||
| # or | ||||
|  | ||||
| docker run traefik[:version] --help | ||||
| # ex: docker run traefik:2.1 --help | ||||
| # ex: docker run traefik:v2.11 --help | ||||
| ``` | ||||
|  | ||||
| All available arguments can also be found [here](../reference/static-configuration/cli.md). | ||||
| Check the [CLI reference](../reference/static-configuration/cli.md "Link to CLI reference overview") for an overview about all available arguments. | ||||
|  | ||||
| ### Environment Variables | ||||
|  | ||||
| All available environment variables can be found [here](../reference/static-configuration/env.md) | ||||
| All available environment variables can be found in the [static configuration environment overview](../reference/static-configuration/env.md). | ||||
|  | ||||
| ## Available Configuration Options | ||||
|  | ||||
| All the configuration options are documented in their related section. | ||||
|  | ||||
| You can browse the available features in the menu, the [providers](../providers/overview.md), or the [routing section](../routing/overview.md) to see them in action. | ||||
|  | ||||
| {!traefik-for-business-applications.md!} | ||||
|   | ||||
							
								
								
									
										253
									
								
								docs/content/getting-started/faq.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										253
									
								
								docs/content/getting-started/faq.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,253 @@ | ||||
| --- | ||||
| title: "Traefik Getting Started FAQ" | ||||
| description: "Check out our FAQ page for answers to commonly asked questions on getting started with Traefik Proxy. Read the technical documentation." | ||||
| --- | ||||
|  | ||||
| # FAQ | ||||
|  | ||||
| ## Why is Traefik Answering `XXX` HTTP Response Status Code? | ||||
|  | ||||
| Traefik is a dynamic reverse proxy, | ||||
| and while the documentation often demonstrates configuration options through file examples, | ||||
| the core feature of Traefik is its dynamic configurability, | ||||
| directly reacting to changes from providers over time. | ||||
|  | ||||
| Notably, a part of the configuration is [static](../configuration-overview/#the-static-configuration), | ||||
| and can be provided by a file on startup, whereas various providers, | ||||
| such as the file provider, | ||||
| contribute dynamically all along the traefik instance lifetime to its [dynamic configuration](../configuration-overview/#the-dynamic-configuration) changes. | ||||
|  | ||||
| In addition, the configuration englobes concepts such as the EntryPoint which can be seen as a listener on the Transport Layer (TCP), | ||||
| as apposed to the Router which is more about the Presentation (TLS) and Application layers (HTTP). | ||||
| And there can be as many routers as one wishes for a given EntryPoint. | ||||
|  | ||||
| In other words, for a given Entrypoint, | ||||
| at any given time the traffic seen is not bound to be just about one protocol. | ||||
| It could be HTTP, or otherwise. Over TLS, or not. | ||||
| Not to mention that dynamic configuration changes potentially make that kind of traffic vary over time. | ||||
|  | ||||
| Therefore, in this dynamic context, | ||||
| the static configuration of an `entryPoint` does not give any hint whatsoever about how the traffic going through that `entryPoint` is going to be routed. | ||||
| Or whether it's even going to be routed at all, | ||||
| that is whether there is a Router matching the kind of traffic going through it. | ||||
|  | ||||
| ### `404 Not found` | ||||
|  | ||||
| Traefik returns a `404` response code in the following situations: | ||||
|  | ||||
| - A request reaching an EntryPoint that has no Routers | ||||
| - An HTTP request reaching an EntryPoint that has no HTTP Router | ||||
| - An HTTPS request reaching an EntryPoint that has no HTTPS Router | ||||
| - A request reaching an EntryPoint that has HTTP/HTTPS Routers that cannot be matched | ||||
|  | ||||
| From Traefik's point of view, | ||||
| every time a request cannot be matched with a router the correct response code is a `404 Not found`. | ||||
|  | ||||
| In this situation, the response code is not a `503 Service Unavailable` | ||||
| because Traefik is not able to confirm that the lack of a matching router for a request is only temporary. | ||||
| Traefik's routing configuration is dynamic and aggregated from different providers, | ||||
| hence it's not possible to assume at any moment that a specific route should be handled or not. | ||||
|  | ||||
| ??? info "This behavior is consistent with rfc7231" | ||||
|  | ||||
|     ```txt | ||||
|     The server is currently unable to handle the request due to a | ||||
|     temporary overloading or maintenance of the server. The implication | ||||
|     is that this is a temporary condition which will be alleviated after | ||||
|     some delay. If known, the length of the delay MAY be indicated in a | ||||
|     Retry-After header. If no Retry-After is given, the client SHOULD | ||||
|     handle the response as it would for a 500 response. | ||||
|  | ||||
|         Note: The existence of the 503 status code does not imply that a | ||||
|         server must use it when becoming overloaded. Some servers may wish | ||||
|         to simply refuse the connection. | ||||
|     ``` | ||||
|  | ||||
|     Extract from [rfc7231#section-6.6.4](https://datatracker.ietf.org/doc/html/rfc7231#section-6.6.4). | ||||
|  | ||||
| ### `502 Bad Gateway` | ||||
|  | ||||
| Traefik returns a `502` response code when an error happens while contacting the upstream service. | ||||
|  | ||||
| ### `503 Service Unavailable` | ||||
|  | ||||
| Traefik returns a `503` response code when a Router has been matched, | ||||
| but there are no servers ready to handle the request. | ||||
|  | ||||
| This situation is encountered when a service has been explicitly configured without servers, | ||||
| or when a service has healthcheck enabled and all servers are unhealthy. | ||||
|  | ||||
| ### `XXX` Instead of `404` | ||||
|  | ||||
| Sometimes, the `404` response code doesn't play well with other parties or services (such as CDNs). | ||||
|  | ||||
| In these situations, you may want Traefik to always reply with a `503` response code, | ||||
| instead of a `404` response code. | ||||
|  | ||||
| To achieve this behavior, a catchall router, | ||||
| with the lowest possible priority and routing to a service without servers, | ||||
| can handle all the requests when no other router has been matched. | ||||
|  | ||||
| The example below is a file provider only version (`yaml`) of what this configuration could look like: | ||||
|  | ||||
| ```yaml tab="Static configuration" | ||||
| # traefik.yml | ||||
|  | ||||
| entrypoints: | ||||
|   web: | ||||
|     address: :80 | ||||
|  | ||||
| providers: | ||||
|   file: | ||||
|     filename: dynamic.yaml | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Dynamic configuration" | ||||
| # dynamic.yaml | ||||
|  | ||||
| http: | ||||
|   routers: | ||||
|     catchall: | ||||
|       # attached only to web entryPoint | ||||
|       entryPoints: | ||||
|         - "web" | ||||
|       # catchall rule | ||||
|       rule: "PathPrefix(`/`)" | ||||
|       service: unavailable | ||||
|       # lowest possible priority | ||||
|       # evaluated when no other router is matched | ||||
|       priority: 1 | ||||
|  | ||||
|   services: | ||||
|     # Service that will always answer a 503 Service Unavailable response | ||||
|     unavailable: | ||||
|       loadBalancer: | ||||
|         servers: {} | ||||
| ``` | ||||
|  | ||||
| !!! info "Dedicated service" | ||||
|     If there is a need for a response code other than a `503` and/or a custom message, | ||||
|     the principle of the above example above (a catchall router) still stands, | ||||
|     but the `unavailable` service should be adapted to fit such a need. | ||||
|  | ||||
| ## Why Is My TLS Certificate Not Reloaded When Its Contents Change? | ||||
|  | ||||
| With the file provider, | ||||
| a configuration update is only triggered when one of the [watched](../providers/file.md#provider-configuration) configuration files is modified. | ||||
|  | ||||
| Which is why, when a certificate is defined by path, | ||||
| and the actual contents of this certificate change, | ||||
| a configuration update is _not_ triggered. | ||||
|  | ||||
| To take into account the new certificate contents, the update of the dynamic configuration must be forced. | ||||
| One way to achieve that, is to trigger a file notification, | ||||
| for example, by using the `touch` command on the configuration file. | ||||
|  | ||||
| ## What Are the Forwarded Headers When Proxying HTTP Requests? | ||||
|  | ||||
| By default, the following headers are automatically added when proxying requests: | ||||
|  | ||||
| | Property                  | HTTP Header                | | ||||
| |---------------------------|----------------------------| | ||||
| | Client's IP               | X-Forwarded-For, X-Real-Ip | | ||||
| | Host                      | X-Forwarded-Host           | | ||||
| | Port                      | X-Forwarded-Port           | | ||||
| | Protocol                  | X-Forwarded-Proto          | | ||||
| | Proxy Server's Hostname   | X-Forwarded-Server         | | ||||
|  | ||||
| For more details, | ||||
| please check out the [forwarded header](../routing/entrypoints.md#forwarded-headers) documentation. | ||||
|  | ||||
| ## How Traefik is Storing and Serving TLS Certificates? | ||||
|  | ||||
| ### Storing TLS Certificates | ||||
|  | ||||
| [TLS](../https/tls.md "Link to Traefik TLS docs") certificates are either provided directly by the [dynamic configuration](./configuration-overview.md#the-dynamic-configuration "Link to dynamic configuration overview") from [providers](../https/tls.md#user-defined "Link to the TLS configuration"), | ||||
| or by [ACME resolvers](../https/acme.md#providers "Link to ACME resolvers"), which act themselves as providers internally. | ||||
|  | ||||
| For each TLS certificate, Traefik produces an identifier used as a key to store it. | ||||
| This identifier is constructed as the alphabetically ordered concatenation of the SANs `DNSNames` and `IPAddresses` of the TLScertificate. | ||||
|  | ||||
| #### Examples: | ||||
|  | ||||
| | X509v3 Subject Alternative Name         | TLS Certificate Identifier  | | ||||
| |-----------------------------------------|-----------------------------| | ||||
| | `DNS:example.com, IP Address:127.0.0.1` | `127.0.0.1,example.com`     | | ||||
| | `DNS:example.com, DNS:*.example.com`    | `*.example.com,example.com` | | ||||
|  | ||||
| The identifier is used to store TLS certificates in order to be later used to handle TLS connections. | ||||
| This operation happens each time there are configuration changes. | ||||
|  | ||||
| If multiple TLS certificates are provided with the same SANs definition (same identifier), only the one processed first is kept. | ||||
| Because the dynamic configuration is aggregated from all providers, | ||||
| when processing it to gather TLS certificates, | ||||
| there is no guarantee of the order in which they would be processed. | ||||
| This means that along with configurations applied, it is possible that the TLS certificate retained for a given identifier differs. | ||||
|  | ||||
| ### Serving TLS Certificates | ||||
|  | ||||
| For each incoming connection, Traefik is serving the "best" matching TLS certificate for the provided server name. | ||||
|  | ||||
| The TLS certificate selection process narrows down the list of TLS certificates matching the server name, | ||||
| and then selects the last TLS certificate in this list after having ordered it by the identifier alphabetically. | ||||
|  | ||||
| #### Examples: | ||||
|  | ||||
| | Selected TLS Certificates Identifiers               | Sorted TLS Certificates Identifiers                 | Served Certificate Identifier | | ||||
| |-----------------------------------------------------|-----------------------------------------------------|-------------------------------| | ||||
| | `127.0.0.1,example.com`,`*.example.com,example.com` | `*.example.com,example.com`,`127.0.0.1,example.com` | `127.0.0.1,example.com`       | | ||||
| | `*.example.com,example.com`,`example.com`           | `*.example.com,example.com`,`example.com`           | `example.com`                 | | ||||
|  | ||||
| ### Caching TLS Certificates | ||||
|  | ||||
| While Traefik is serving the best matching TLS certificate for each incoming connection, | ||||
| the selection process cost for each incoming connection is avoided thanks to a cache mechanism. | ||||
|  | ||||
| Once a TLS certificate has been selected as the "best" TLS certificate for a server name, | ||||
| it is cached for an hour, avoiding the selection process for further connections. | ||||
|  | ||||
| Nonetheless, when a new configuration is applied, the cache is reset. | ||||
|  | ||||
| ## What does the "field not found" error mean? | ||||
|  | ||||
| ```shell | ||||
| error: field not found, node: -badField- | ||||
| ``` | ||||
|  | ||||
| The "field not found" error occurs, when an unknown property is encountered in the dynamic or static configuration. | ||||
|  | ||||
| One way to check whether a configuration file is well-formed, is to validate it with: | ||||
|  | ||||
| - [JSON Schema of the static configuration](https://json.schemastore.org/traefik-v2.json) | ||||
| - [JSON Schema of the dynamic configuration](https://json.schemastore.org/traefik-v2-file-provider.json) | ||||
|  | ||||
| ## Why are some resources (routers, middlewares, services...) not created/applied? | ||||
|  | ||||
| As a common tip, if a resource is dropped/not created by Traefik after the dynamic configuration was evaluated, | ||||
| one should look for an error in the logs. | ||||
|  | ||||
| If found, the error confirms that something went wrong while creating the resource, | ||||
| and the message should help in figuring out the mistake(s) in the configuration, and how to fix it. | ||||
|  | ||||
| When using the file provider, | ||||
| one way to check if the dynamic configuration is well-formed is to validate it with the [JSON Schema of the dynamic configuration](https://json.schemastore.org/traefik-v2-file-provider.json). | ||||
|  | ||||
| ## Why does Let's Encrypt wildcard certificate renewal/generation with DNS challenge fail? | ||||
|  | ||||
| If you're trying to renew wildcard certificates, with DNS challenge, | ||||
| and you're getting errors such as: | ||||
|  | ||||
| ```txt | ||||
| msg="Error renewing certificate from LE: {example.com [*.example.com]}" | ||||
| providerName=letsencrypt.acme error="error: one or more domains had a problem: | ||||
| [example.com] acme: error presenting token: gandiv5: unexpected authZone example.com. for fqdn example.com." | ||||
| ``` | ||||
|  | ||||
| then it could be due to `CNAME` support. | ||||
|  | ||||
| In which case, you should make sure your infrastructure is properly set up for a | ||||
| `DNS` challenge that does not rely on `CNAME`, and you should try disabling `CNAME` support with: | ||||
|  | ||||
| ```shell | ||||
| LEGO_DISABLE_CNAME_SUPPORT=true | ||||
| ``` | ||||
| @@ -1,3 +1,8 @@ | ||||
| --- | ||||
| title: "Traefik Installation Documentation" | ||||
| description: "There are several flavors to choose from when installing Traefik Proxy. Get started with Traefik Proxy, and read the technical documentation." | ||||
| --- | ||||
|  | ||||
| # Install Traefik | ||||
|  | ||||
| You can install Traefik with the following flavors: | ||||
| @@ -9,11 +14,14 @@ You can install Traefik with the following flavors: | ||||
|  | ||||
| ## Use the Official Docker Image | ||||
|  | ||||
| Choose one of the [official Docker images](https://hub.docker.com/_/traefik) and run it with the [sample configuration file](https://raw.githubusercontent.com/containous/traefik/v2.2/traefik.sample.toml): | ||||
| Choose one of the [official Docker images](https://hub.docker.com/_/traefik) and run it with one sample configuration file: | ||||
|  | ||||
| ```bash | ||||
| * [YAML](https://raw.githubusercontent.com/traefik/traefik/v2.11/traefik.sample.yml) | ||||
| * [TOML](https://raw.githubusercontent.com/traefik/traefik/v2.11/traefik.sample.toml) | ||||
|  | ||||
| ```shell | ||||
| docker run -d -p 8080:8080 -p 80:80 \ | ||||
|     -v $PWD/traefik.toml:/etc/traefik/traefik.toml traefik:v2.2 | ||||
|     -v $PWD/traefik.yml:/etc/traefik/traefik.yml traefik:v2.11 | ||||
| ``` | ||||
|  | ||||
| For more details, go to the [Docker provider documentation](../providers/docker.md) | ||||
| @@ -21,28 +29,28 @@ For more details, go to the [Docker provider documentation](../providers/docker. | ||||
| !!! tip | ||||
|  | ||||
|     * Prefer a fixed version than the latest that could be an unexpected version. | ||||
|     ex: `traefik:v2.1.4` | ||||
|     ex: `traefik:v2.11` | ||||
|     * Docker images are based from the [Alpine Linux Official image](https://hub.docker.com/_/alpine). | ||||
|     * Any orchestrator using docker images can fetch the official Traefik docker image. | ||||
|  | ||||
| ## Use the Helm Chart | ||||
|  | ||||
| !!! warning | ||||
|      | ||||
|     The Traefik Chart from  | ||||
|     [Helm's default charts repository](https://github.com/helm/charts/tree/master/stable/traefik) is still using [Traefik v1.7](https://docs.traefik.io/v1.7). | ||||
|  | ||||
| Traefik can be installed in Kubernetes using the Helm chart from <https://github.com/containous/traefik-helm-chart>. | ||||
|     The Traefik Chart from | ||||
|     [Helm's default charts repository](https://github.com/helm/charts/tree/master/stable/traefik) is still using [Traefik v1.7](https://doc.traefik.io/traefik/v1.7). | ||||
|  | ||||
| Traefik can be installed in Kubernetes using the Helm chart from <https://github.com/traefik/traefik-helm-chart>. | ||||
|  | ||||
| Ensure that the following requirements are met: | ||||
|  | ||||
| * Kubernetes 1.14+ | ||||
| * Helm version 3.x is [installed](https://helm.sh/docs/intro/install/) | ||||
| * Kubernetes 1.16+ | ||||
| * Helm version 3.9+ is [installed](https://helm.sh/docs/intro/install/) | ||||
|  | ||||
| Add Traefik's chart repository to Helm: | ||||
| Add Traefik Labs chart repository to Helm: | ||||
|  | ||||
| ```bash | ||||
| helm repo add traefik https://containous.github.io/traefik-helm-chart | ||||
| helm repo add traefik https://traefik.github.io/charts | ||||
| ``` | ||||
|  | ||||
| You can update the chart repository by running: | ||||
| @@ -51,15 +59,18 @@ You can update the chart repository by running: | ||||
| helm repo update | ||||
| ``` | ||||
|  | ||||
| And install it with the `helm` command line: | ||||
| And install it with the Helm command line: | ||||
|  | ||||
| ```bash | ||||
| helm install traefik traefik/traefik | ||||
| ``` | ||||
|  | ||||
| !!! tip "Helm Features" | ||||
|      | ||||
|  | ||||
|     All [Helm features](https://helm.sh/docs/intro/using_helm/) are supported. | ||||
|  | ||||
|     Examples are provided [here](https://github.com/traefik/traefik-helm-chart/blob/master/EXAMPLES.md). | ||||
|  | ||||
|     For instance, installing the chart in a dedicated namespace: | ||||
|  | ||||
|     ```bash tab="Install in a Dedicated Namespace" | ||||
| @@ -70,48 +81,47 @@ helm install traefik traefik/traefik | ||||
|     ``` | ||||
|  | ||||
| ??? example "Installing with Custom Values" | ||||
|      | ||||
|  | ||||
|     You can customize the installation by specifying custom values, | ||||
|     as with [any helm chart](https://helm.sh/docs/intro/using_helm/#customizing-the-chart-before-installing). | ||||
|     {: #helm-custom-values } | ||||
|      | ||||
|     The values are not (yet) documented, but are self-explanatory: | ||||
|     you can look at the [default `values.yaml`](https://github.com/containous/traefik-helm-chart/blob/master/traefik/values.yaml) file to explore possibilities. | ||||
|      | ||||
|  | ||||
|     All parameters are documented in the default [`values.yaml`](https://github.com/traefik/traefik-helm-chart/blob/master/traefik/values.yaml). | ||||
|  | ||||
|     You can also set Traefik command line flags using `additionalArguments`. | ||||
|     Example of installation with logging set to `DEBUG`: | ||||
|      | ||||
|  | ||||
|     ```bash tab="Using Helm CLI" | ||||
|     helm install --namespace=traefik-v2 \ | ||||
|         --set="additionalArguments={--log.level=DEBUG}" \ | ||||
|         traefik traefik/traefik | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```yml tab="With a custom values file" | ||||
|     # File custom-values.yml | ||||
|     ## Install with "helm install --values=./custom-values.yml traefik traefik/traefik | ||||
|     additionalArguments: | ||||
|       - "--log.level=DEBUG" | ||||
|     ``` | ||||
|      | ||||
|  | ||||
| ### Exposing the Traefik dashboard | ||||
|  | ||||
| This HelmChart does not expose the Traefik dashboard by default, for security concerns. | ||||
| This Helm chart does not expose the Traefik dashboard by default, for security concerns. | ||||
| Thus, there are multiple ways to expose the dashboard. | ||||
| For instance, the dashboard access could be achieved through a port-forward : | ||||
| For instance, the dashboard access could be achieved through a port-forward: | ||||
|  | ||||
| ```shell | ||||
| kubectl port-forward $(kubectl get pods --selector "app.kubernetes.io/name=traefik" --output=name) 9000:9000 | ||||
| ``` | ||||
|  | ||||
| Accessible with the url: http://127.0.0.1:9000/dashboard/ | ||||
| It can then be reached at: `http://127.0.0.1:9000/dashboard/` | ||||
|  | ||||
| Another way would be to apply your own configuration, for instance, | ||||
| by defining and applying an IngressRoute CRD (`kubectl apply -f dashboard.yaml`): | ||||
|  | ||||
| ```yaml | ||||
| # dashboard.yaml | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: IngressRoute | ||||
| metadata: | ||||
|   name: dashboard | ||||
| @@ -128,7 +138,7 @@ spec: | ||||
|  | ||||
| ## Use the Binary Distribution | ||||
|  | ||||
| Grab the latest binary from the [releases](https://github.com/containous/traefik/releases) page. | ||||
| Grab the latest binary from the [releases](https://github.com/traefik/traefik/releases) page. | ||||
|  | ||||
| ??? info "Check the integrity of the downloaded file" | ||||
|  | ||||
| @@ -170,3 +180,5 @@ And run it: | ||||
| ## Compile your Binary from the Sources | ||||
|  | ||||
| All the details are available in the [Contributing Guide](../contributing/building-testing.md) | ||||
|  | ||||
| {!traefik-for-business-applications.md!} | ||||
|   | ||||
							
								
								
									
										320
									
								
								docs/content/getting-started/quick-start-with-kubernetes.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										320
									
								
								docs/content/getting-started/quick-start-with-kubernetes.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,320 @@ | ||||
| --- | ||||
| title: "Traefik Getting Started With Kubernetes" | ||||
| description: "Get started with Traefik Proxy and Kubernetes." | ||||
| --- | ||||
|  | ||||
| # Quick Start | ||||
|  | ||||
| A Use Case of Traefik Proxy and Kubernetes | ||||
| {: .subtitle } | ||||
|  | ||||
| This guide is an introduction to using Traefik Proxy in a Kubernetes environment.   | ||||
| The objective is to learn how to run an application behind a Traefik reverse proxy in Kubernetes.   | ||||
| It presents and explains the basic blocks required to start with Traefik such as Ingress Controller, Ingresses, Deployments, static, and dynamic configuration. | ||||
|  | ||||
| ## Permissions and Accesses | ||||
|  | ||||
| Traefik uses the Kubernetes API to discover running services. | ||||
|  | ||||
| To use the Kubernetes API, Traefik needs some permissions. | ||||
| This [permission mechanism](https://kubernetes.io/docs/reference/access-authn-authz/rbac/) is based on roles defined by the cluster administrator.   | ||||
| The role is then bound to an account used by an application, in this case, Traefik Proxy. | ||||
|  | ||||
| The first step is to create the role. | ||||
| The [`ClusterRole`](https://kubernetes.io/docs/reference/kubernetes-api/authorization-resources/cluster-role-v1/#ClusterRole) resource enumerates the resources and actions available for the role. | ||||
| In a file called `00-role.yml`, put the following `ClusterRole`: | ||||
|  | ||||
| ```yaml tab="00-role.yml" | ||||
| kind: ClusterRole | ||||
| apiVersion: rbac.authorization.k8s.io/v1 | ||||
| metadata: | ||||
|   name: traefik-role | ||||
|  | ||||
| rules: | ||||
|   - apiGroups: | ||||
|       - "" | ||||
|     resources: | ||||
|       - services | ||||
|       - endpoints | ||||
|       - secrets | ||||
|     verbs: | ||||
|       - get | ||||
|       - list | ||||
|       - watch | ||||
|   - apiGroups: | ||||
|       - extensions | ||||
|       - networking.k8s.io | ||||
|     resources: | ||||
|       - ingresses | ||||
|       - ingressclasses | ||||
|     verbs: | ||||
|       - get | ||||
|       - list | ||||
|       - watch | ||||
|   - apiGroups: | ||||
|       - extensions | ||||
|       - networking.k8s.io | ||||
|     resources: | ||||
|       - ingresses/status | ||||
|     verbs: | ||||
|       - update | ||||
| ``` | ||||
|  | ||||
| !!! info "You can find the reference for this file [there](../../reference/dynamic-configuration/kubernetes-crd/#rbac)." | ||||
|  | ||||
| The next step is to create a dedicated service account for Traefik. | ||||
| In a file called `00-account.yml`, put the following [`ServiceAccount`](https://kubernetes.io/docs/reference/kubernetes-api/authentication-resources/service-account-v1/#ServiceAccount) resource: | ||||
|  | ||||
| ```yaml tab="00-account.yml" | ||||
| apiVersion: v1 | ||||
| kind: ServiceAccount | ||||
| metadata: | ||||
|   name: traefik-account | ||||
| ``` | ||||
|  | ||||
| And then, bind the role on the account to apply the permissions and rules on the latter. In a file called `01-role-binding.yml`, put the | ||||
| following [`ClusterRoleBinding`](https://kubernetes.io/docs/reference/kubernetes-api/authorization-resources/cluster-role-binding-v1/#ClusterRoleBinding) resource: | ||||
|  | ||||
| ```yaml tab="01-role-binding.yml" | ||||
| kind: ClusterRoleBinding | ||||
| apiVersion: rbac.authorization.k8s.io/v1 | ||||
| metadata: | ||||
|   name: traefik-role-binding | ||||
|  | ||||
| roleRef: | ||||
|   apiGroup: rbac.authorization.k8s.io | ||||
|   kind: ClusterRole | ||||
|   name: traefik-role | ||||
| subjects: | ||||
|   - kind: ServiceAccount | ||||
|     name: traefik-account | ||||
|     namespace: default # This tutorial uses the "default" K8s namespace. | ||||
| ``` | ||||
|  | ||||
| !!! info "`roleRef` is the Kubernetes reference to the role created in `00-role.yml`." | ||||
|  | ||||
| !!! info "`subjects` is the list of accounts reference." | ||||
|  | ||||
|     In this guide, it only contains the account created in `00-account.yml` | ||||
|  | ||||
| ## Deployment and Exposition | ||||
|  | ||||
| !!! info "This section can be managed with the help of the [Traefik Helm chart](../install-traefik/#use-the-helm-chart)." | ||||
|  | ||||
| The [ingress controller](https://traefik.io/glossary/kubernetes-ingress-and-ingress-controller-101/#what-is-a-kubernetes-ingress-controller) | ||||
| is a software that runs in the same way as any other application on a cluster.   | ||||
| To start Traefik on the Kubernetes cluster, | ||||
| a [`Deployment`](https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/deployment-v1/) resource must exist to describe how to configure | ||||
| and scale containers horizontally to support larger workloads. | ||||
|  | ||||
| Start by creating a file called `02-traefik.yml` and paste the following `Deployment` resource: | ||||
|  | ||||
| ```yaml tab="02-traefik.yml" | ||||
| kind: Deployment | ||||
| apiVersion: apps/v1 | ||||
| metadata: | ||||
|   name: traefik-deployment | ||||
|   labels: | ||||
|     app: traefik | ||||
|  | ||||
| spec: | ||||
|   replicas: 1 | ||||
|   selector: | ||||
|     matchLabels: | ||||
|       app: traefik | ||||
|   template: | ||||
|     metadata: | ||||
|       labels: | ||||
|         app: traefik | ||||
|     spec: | ||||
|       serviceAccountName: traefik-account | ||||
|       containers: | ||||
|         - name: traefik | ||||
|           image: traefik:v2.11 | ||||
|           args: | ||||
|             - --api.insecure | ||||
|             - --providers.kubernetesingress | ||||
|           ports: | ||||
|             - name: web | ||||
|               containerPort: 80 | ||||
|             - name: dashboard | ||||
|               containerPort: 8080 | ||||
| ``` | ||||
|  | ||||
| The deployment contains an important attribute for customizing Traefik: `args`.   | ||||
| These arguments are the static configuration for Traefik.   | ||||
| From here, it is possible to enable the dashboard, | ||||
| configure entry points, | ||||
| select dynamic configuration providers, | ||||
| and [more](../reference/static-configuration/cli.md). | ||||
|  | ||||
| In this deployment, | ||||
| the static configuration enables the Traefik dashboard, | ||||
| and uses Kubernetes native Ingress resources as router definitions to route incoming requests. | ||||
|  | ||||
| !!! info "When there is no entry point in the static configuration" | ||||
|  | ||||
|     Traefik creates a default one called `web` using the port `80` routing HTTP requests. | ||||
|  | ||||
| !!! info "When enabling the [`api.insecure`](../../operations/api/#insecure) mode, Traefik exposes the dashboard on the port `8080`." | ||||
|  | ||||
| A deployment manages scaling and then can create lots of containers, called [Pods](https://kubernetes.io/docs/concepts/workloads/pods/). | ||||
| Each Pod is configured following the `spec` field in the deployment.   | ||||
| Given that, a Deployment can run multiple Traefik Proxy Pods, | ||||
| a piece is required to forward the traffic to any of the instance: | ||||
| namely a [`Service`](https://kubernetes.io/docs/reference/kubernetes-api/service-resources/service-v1/#Service).   | ||||
| Create a file called `02-traefik-services.yml` and insert the two `Service` resources: | ||||
|  | ||||
| ```yaml tab="02-traefik-services.yml" | ||||
| apiVersion: v1 | ||||
| kind: Service | ||||
| metadata: | ||||
|   name: traefik-dashboard-service | ||||
|  | ||||
| spec: | ||||
|   type: LoadBalancer | ||||
|   ports: | ||||
|     - port: 8080 | ||||
|       targetPort: dashboard | ||||
|   selector: | ||||
|     app: traefik | ||||
| --- | ||||
| apiVersion: v1 | ||||
| kind: Service | ||||
| metadata: | ||||
|   name: traefik-web-service | ||||
|  | ||||
| spec: | ||||
|   type: LoadBalancer | ||||
|   ports: | ||||
|     - targetPort: web | ||||
|       port: 80 | ||||
|   selector: | ||||
|     app: traefik | ||||
| ``` | ||||
|  | ||||
| !!! warning "It is possible to expose a service in different ways." | ||||
|  | ||||
|     Depending on your working environment and use case, the `spec.type` might change.   | ||||
|     It is strongly recommended to understand the available [service types](https://kubernetes.io/docs/concepts/services-networking/service/#publishing-services-service-types) before proceeding to the next step. | ||||
|  | ||||
| It is now time to apply those files on your cluster to start Traefik. | ||||
|  | ||||
| ```shell | ||||
| kubectl apply -f 00-role.yml \ | ||||
|               -f 00-account.yml \ | ||||
|               -f 01-role-binding.yml \ | ||||
|               -f 02-traefik.yml \ | ||||
|               -f 02-traefik-services.yml | ||||
| ``` | ||||
|  | ||||
| ## Proxying applications | ||||
|  | ||||
| The only part still missing is the business application behind the reverse proxy.   | ||||
| For this guide, we use the example application [traefik/whoami](https://github.com/traefik/whoami), | ||||
| but the principles are applicable to any other application. | ||||
|  | ||||
| The `whoami` application is an HTTP server running on port 80 which answers host-related information to the incoming requests.   | ||||
| As usual, start by creating a file called `03-whoami.yml` and paste the following `Deployment` resource: | ||||
|  | ||||
| ```yaml tab="03-whoami.yml" | ||||
| kind: Deployment | ||||
| apiVersion: apps/v1 | ||||
| metadata: | ||||
|   name: whoami | ||||
|   labels: | ||||
|     app: whoami | ||||
|  | ||||
| spec: | ||||
|   replicas: 1 | ||||
|   selector: | ||||
|     matchLabels: | ||||
|       app: whoami | ||||
|   template: | ||||
|     metadata: | ||||
|       labels: | ||||
|         app: whoami | ||||
|     spec: | ||||
|       containers: | ||||
|         - name: whoami | ||||
|           image: traefik/whoami | ||||
|           ports: | ||||
|             - name: web | ||||
|               containerPort: 80 | ||||
| ``` | ||||
|  | ||||
| And continue by creating the following `Service` resource in a file called `03-whoami-services.yml`: | ||||
|  | ||||
| ```yaml tab="03-whoami-services.yml" | ||||
| apiVersion: v1 | ||||
| kind: Service | ||||
| metadata: | ||||
|   name: whoami | ||||
|  | ||||
| spec: | ||||
|   ports: | ||||
|     - name: web | ||||
|       port: 80 | ||||
|       targetPort: web | ||||
|        | ||||
|   selector: | ||||
|     app: whoami | ||||
| ``` | ||||
|  | ||||
| Thanks to the Kubernetes API, | ||||
| Traefik is notified when an Ingress resource is created, updated, or deleted.   | ||||
| This makes the process dynamic.   | ||||
| The ingresses are, in a way, the [dynamic configuration](../../providers/kubernetes-ingress/) for Traefik. | ||||
|  | ||||
| !!! tip | ||||
|  | ||||
|     Find more information on [ingress controller](https://kubernetes.io/docs/concepts/services-networking/ingress-controllers/), | ||||
|     and [Ingress](https://kubernetes.io/docs/concepts/services-networking/ingress/) in the official Kubernetes documentation. | ||||
|  | ||||
| Create a file called `04-whoami-ingress.yml` and insert the `Ingress` resource: | ||||
|  | ||||
| ```yaml tab="04-whoami-ingress.yml" | ||||
| apiVersion: networking.k8s.io/v1 | ||||
| kind: Ingress | ||||
| metadata: | ||||
|   name: whoami-ingress | ||||
| spec: | ||||
|   rules: | ||||
|   - http: | ||||
|       paths: | ||||
|       - path: / | ||||
|         pathType: Prefix | ||||
|         backend: | ||||
|           service: | ||||
|             name: whoami | ||||
|             port: | ||||
|               name: web | ||||
| ``` | ||||
|  | ||||
| This `Ingress` configures Traefik to redirect any incoming requests starting with `/` to the `whoami:80` service. | ||||
|  | ||||
| At this point, all the configurations are ready. | ||||
| It is time to apply those new files: | ||||
|  | ||||
| ```shell | ||||
| kubectl apply -f 03-whoami.yml \ | ||||
|               -f 03-whoami-services.yml \ | ||||
|               -f 04-whoami-ingress.yml | ||||
| ``` | ||||
|  | ||||
| Now you should be able to access the `whoami` application and the Traefik dashboard. | ||||
| Load the dashboard on a web browser: [`http://localhost:8080`](http://localhost:8080). | ||||
|  | ||||
| And now access the `whoami` application: | ||||
|  | ||||
| ```shell | ||||
| curl -v http://localhost/ | ||||
| ``` | ||||
|  | ||||
| !!! question "Going further" | ||||
|  | ||||
|     - [Filter the ingresses](../providers/kubernetes-ingress.md#ingressclass) to use with [IngressClass](https://kubernetes.io/docs/concepts/services-networking/ingress/#ingress-class) | ||||
|     - Use [IngressRoute CRD](../providers/kubernetes-crd.md) | ||||
|     - Protect [ingresses with TLS](../routing/providers/kubernetes-ingress.md#enabling-tls-via-annotations) | ||||
|      | ||||
| {!traefik-for-business-applications.md!} | ||||
| @@ -1,6 +1,11 @@ | ||||
| --- | ||||
| title: "Traefik Getting Started Quickly" | ||||
| description: "Get started with Traefik Proxy and Docker." | ||||
| --- | ||||
|  | ||||
| # Quick Start | ||||
|  | ||||
| A Simple Use Case Using Docker | ||||
| A Use Case Using Docker | ||||
| {: .subtitle } | ||||
|  | ||||
|  | ||||
| @@ -15,7 +20,7 @@ version: '3' | ||||
| services: | ||||
|   reverse-proxy: | ||||
|     # The official v2 Traefik docker image | ||||
|     image: traefik:v2.2 | ||||
|     image: traefik:v2.11 | ||||
|     # Enables the web UI and tells Traefik to listen to docker | ||||
|     command: --api.insecure=true --providers.docker | ||||
|     ports: | ||||
| @@ -36,24 +41,29 @@ Start your `reverse-proxy` with the following command: | ||||
| docker-compose up -d reverse-proxy | ||||
| ``` | ||||
|  | ||||
| You can open a browser and go to [http://localhost:8080/api/rawdata](http://localhost:8080/api/rawdata) to see Traefik's API rawdata (we'll go back there once we have launched a service in step 2). | ||||
| You can open a browser and go to `http://localhost:8080/api/rawdata` to see Traefik's API rawdata (you'll go back there once you have launched a service in step 2). | ||||
|  | ||||
| ## Traefik Detects New Services and Creates the Route for You | ||||
|  | ||||
| Now that we have a Traefik instance up and running, we will deploy new services. | ||||
| Now that you have a Traefik instance up and running, you will deploy new services. | ||||
|  | ||||
| Edit your `docker-compose.yml` file and add the following at the end of your file. | ||||
|  | ||||
| ```yaml | ||||
| # ... | ||||
| version: '3' | ||||
|  | ||||
| services: | ||||
|  | ||||
|   ... | ||||
|  | ||||
|   whoami: | ||||
|     # A container that exposes an API to show its IP address | ||||
|     image: containous/whoami | ||||
|     image: traefik/whoami | ||||
|     labels: | ||||
|       - "traefik.http.routers.whoami.rule=Host(`whoami.docker.localhost`)" | ||||
| ``` | ||||
|  | ||||
| The above defines `whoami`: a simple web service that outputs information about the machine it is deployed on (its IP address, host, and so on). | ||||
| The above defines `whoami`: a web service that outputs information about the machine it is deployed on (its IP address, host, and others). | ||||
|  | ||||
| Start the `whoami` service with the following command: | ||||
|  | ||||
| @@ -61,9 +71,9 @@ Start the `whoami` service with the following command: | ||||
| docker-compose up -d whoami | ||||
| ``` | ||||
|  | ||||
| Go back to your browser ([http://localhost:8080/api/rawdata](http://localhost:8080/api/rawdata)) and see that Traefik has automatically detected the new container and updated its own configuration. | ||||
| Go back to your browser (`http://localhost:8080/api/rawdata`) and see that Traefik has automatically detected the new container and updated its own configuration. | ||||
|  | ||||
| When Traefik detects new services, it creates the corresponding routes so you can call them ... _let's see!_  (Here, we're using curl) | ||||
| When Traefik detects new services, it creates the corresponding routes so you can call them ... _let's see!_  (Here, you're using curl) | ||||
|  | ||||
| ```shell | ||||
| curl -H Host:whoami.docker.localhost http://127.0.0.1 | ||||
| @@ -85,7 +95,7 @@ Run more instances of your `whoami` service with the following command: | ||||
| docker-compose up -d --scale whoami=2 | ||||
| ``` | ||||
|  | ||||
| Go back to your browser ([http://localhost:8080/api/rawdata](http://localhost:8080/api/rawdata)) and see that Traefik has automatically detected the new instance of the container. | ||||
| Go back to your browser (`http://localhost:8080/api/rawdata`) and see that Traefik has automatically detected the new instance of the container. | ||||
|  | ||||
| Finally, see that Traefik load-balances between the two instances of your service by running the following command twice: | ||||
|  | ||||
| @@ -93,7 +103,7 @@ Finally, see that Traefik load-balances between the two instances of your servic | ||||
| curl -H Host:whoami.docker.localhost http://127.0.0.1 | ||||
| ``` | ||||
|  | ||||
| The output will show alternatively one of the followings: | ||||
| The output will show alternatively one of the following: | ||||
|  | ||||
| ```yaml | ||||
| Hostname: a656c8ddca6c | ||||
| @@ -108,4 +118,7 @@ IP: 172.27.0.4 | ||||
| ``` | ||||
|  | ||||
| !!! question "Where to Go Next?" | ||||
|  | ||||
|     Now that you have a basic understanding of how Traefik can automatically create the routes to your services and load balance them, it is time to dive into [the documentation](/) and let Traefik work for you! | ||||
|  | ||||
| {!traefik-for-business-applications.md!} | ||||
|   | ||||
| @@ -1,22 +0,0 @@ | ||||
| # TODO -- Glossary | ||||
|  | ||||
| Where Every Technical Word finds its Definition` | ||||
| {: .subtitle} | ||||
|  | ||||
| - [ ] Provider | ||||
|     - [ ] Types of providers (KV, annotation based, label based, configuration based) | ||||
| - [ ] Entrypoint | ||||
| - [ ] Routers | ||||
| - [ ] Middleware | ||||
| - [ ] Service | ||||
| - [ ] [Static configuration](getting-started/configuration-overview.md#the-static-configuration) | ||||
| - [ ] [Dynamic configuration](getting-started/configuration-overview.md#the-dynamic-configuration) | ||||
| - [ ] ACME | ||||
| - [ ] TraefikEE | ||||
| - [ ] Tracing | ||||
| - [ ] Metrics | ||||
| - [ ] Orchestrator | ||||
| - [ ] Key Value Store | ||||
| - [ ] Logs | ||||
| - [ ] Traefiker | ||||
| - [ ] Traefik (How to pronounce) | ||||
| @@ -1,3 +1,8 @@ | ||||
| --- | ||||
| title: "Traefik Let's Encrypt Documentation" | ||||
| description: "Learn how to configure Traefik Proxy to use an ACME provider like Let's Encrypt for automatic certificate generation. Read the technical documentation." | ||||
| --- | ||||
|  | ||||
| # Let's Encrypt | ||||
|  | ||||
| Automatic HTTPS | ||||
| @@ -6,14 +11,18 @@ Automatic HTTPS | ||||
| You can configure Traefik to use an ACME provider (like Let's Encrypt) for automatic certificate generation. | ||||
|  | ||||
| !!! warning "Let's Encrypt and Rate Limiting" | ||||
|     Note that Let's Encrypt API has [rate limiting](https://letsencrypt.org/docs/rate-limits). | ||||
|     Note that Let's Encrypt API has [rate limiting](https://letsencrypt.org/docs/rate-limits). These last up to **one week**, and can not be overridden. | ||||
|      | ||||
|     When running Traefik in a container this file should be persisted across restarts.  | ||||
|     If Traefik requests new certificates each time it starts up, a crash-looping container can quickly reach Let's Encrypt's ratelimits. | ||||
|     To configure where certificates are stored, please take a look at the [storage](#storage) configuration. | ||||
|  | ||||
|     Use Let's Encrypt staging server with the [`caServer`](#caserver) configuration option | ||||
|     when experimenting to avoid hitting this limit too fast. | ||||
|      | ||||
|  | ||||
| ## Certificate Resolvers | ||||
|  | ||||
| Traefik requires you to define "Certificate Resolvers" in the [static configuration](../getting-started/configuration-overview.md#the-static-configuration),  | ||||
| Traefik requires you to define "Certificate Resolvers" in the [static configuration](../getting-started/configuration-overview.md#the-static-configuration), | ||||
| which are responsible for retrieving certificates from an ACME server. | ||||
|  | ||||
| Then, each ["router"](../routing/routers/index.md) is configured to enable TLS, | ||||
| @@ -23,36 +32,38 @@ Certificates are requested for domain names retrieved from the router's [dynamic | ||||
|  | ||||
| You can read more about this retrieval mechanism in the following section: [ACME Domain Definition](#domain-definition). | ||||
|  | ||||
| !!! important "Defining a certificates resolver does not result in all routers automatically using it. Each router that is supposed to use the resolver must [reference](../routing/routers/index.md#certresolver) it." | ||||
| !!! warning "Defining an [ACME challenge type](#the-different-acme-challenges) is a requirement for a certificate resolver to be functional." | ||||
|  | ||||
| !!! important "Defining a certificate resolver does not result in all routers automatically using it. Each router that is supposed to use the resolver must [reference](../routing/routers/index.md#certresolver) it." | ||||
|  | ||||
| ??? note "Configuration Reference" | ||||
|      | ||||
|  | ||||
|     There are many available options for ACME. | ||||
|     For a quick glance at what's possible, browse the configuration reference: | ||||
|      | ||||
|     ```toml tab="File (TOML)" | ||||
|     --8<-- "content/https/ref-acme.toml" | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```yaml tab="File (YAML)" | ||||
|     --8<-- "content/https/ref-acme.yaml" | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     --8<-- "content/https/ref-acme.toml" | ||||
|     ``` | ||||
|  | ||||
|     ```bash tab="CLI" | ||||
|     --8<-- "content/https/ref-acme.txt" | ||||
|     ``` | ||||
|  | ||||
| ## Domain Definition | ||||
|  | ||||
| Certificate resolvers request certificates for a set of the domain names  | ||||
| Certificate resolvers request certificates for a set of the domain names | ||||
| inferred from routers, with the following logic: | ||||
|  | ||||
| - If the router has a [`tls.domains`](../routing/routers/index.md#domains) option set, | ||||
|   then the certificate resolver uses the `main` (and optionally `sans`) option of `tls.domains` to know the domain names for this router. | ||||
|  | ||||
| - If no [`tls.domains`](../routing/routers/index.md#domains) option is set,  | ||||
|   then the certificate resolver uses the [router's rule](../routing/routers/index.md#rule),  | ||||
|   by checking the `Host()` matchers.  | ||||
| - If no [`tls.domains`](../routing/routers/index.md#domains) option is set, | ||||
|   then the certificate resolver uses the [router's rule](../routing/routers/index.md#rule), | ||||
|   by checking the `Host()` matchers. | ||||
|   Please note that [multiple `Host()` matchers can be used](../routing/routers/index.md#certresolver)) for specifying multiple domain names for this router. | ||||
|  | ||||
| Please note that: | ||||
| @@ -69,31 +80,15 @@ Please check the [configuration examples below](#configuration-examples) for mor | ||||
| ## Configuration Examples | ||||
|  | ||||
| ??? example "Enabling ACME" | ||||
|      | ||||
|     ```toml tab="File (TOML)" | ||||
|     [entryPoints] | ||||
|       [entryPoints.web] | ||||
|         address = ":80" | ||||
|      | ||||
|       [entryPoints.websecure] | ||||
|         address = ":443" | ||||
|      | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       email = "your-email@example.com" | ||||
|       storage = "acme.json" | ||||
|       [certificatesResolvers.myresolver.acme.httpChallenge] | ||||
|         # used during the challenge | ||||
|         entryPoint = "web" | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```yaml tab="File (YAML)" | ||||
|     entryPoints: | ||||
|       web: | ||||
|         address: ":80" | ||||
|      | ||||
|  | ||||
|       websecure: | ||||
|         address: ":443" | ||||
|      | ||||
|  | ||||
|     certificatesResolvers: | ||||
|       myresolver: | ||||
|         acme: | ||||
| @@ -103,7 +98,23 @@ Please check the [configuration examples below](#configuration-examples) for mor | ||||
|             # used during the challenge | ||||
|             entryPoint: web | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [entryPoints] | ||||
|       [entryPoints.web] | ||||
|         address = ":80" | ||||
|  | ||||
|       [entryPoints.websecure] | ||||
|         address = ":443" | ||||
|  | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       email = "your-email@example.com" | ||||
|       storage = "acme.json" | ||||
|       [certificatesResolvers.myresolver.acme.httpChallenge] | ||||
|         # used during the challenge | ||||
|         entryPoint = "web" | ||||
|     ``` | ||||
|  | ||||
|     ```bash tab="CLI" | ||||
|     --entrypoints.web.address=:80 | ||||
|     --entrypoints.websecure.address=:443 | ||||
| @@ -114,33 +125,37 @@ Please check the [configuration examples below](#configuration-examples) for mor | ||||
|     --certificatesresolvers.myresolver.acme.httpchallenge.entrypoint=web | ||||
|     ``` | ||||
|  | ||||
| !!! important "Defining a certificates resolver does not result in all routers automatically using it. Each router that is supposed to use the resolver must [reference](../routing/routers/index.md#certresolver) it." | ||||
| !!! important "Defining a certificate resolver does not result in all routers automatically using it. Each router that is supposed to use the resolver must [reference](../routing/routers/index.md#certresolver) it." | ||||
|  | ||||
| ??? example "Single Domain from Router's Rule Example" | ||||
|      | ||||
|  | ||||
|     * A certificate for the domain `example.com` is requested: | ||||
|  | ||||
|     --8<-- "content/https/include-acme-single-domain-example.md" | ||||
|  | ||||
| ??? example "Multiple Domains from Router's Rule Example" | ||||
|   | ||||
|  | ||||
|     * A certificate for the domains `example.com` (main) and `blog.example.org` | ||||
|       is requested: | ||||
|      | ||||
|  | ||||
|     --8<-- "content/https/include-acme-multiple-domains-from-rule-example.md" | ||||
|      | ||||
|  | ||||
| ??? example "Multiple Domains from Router's `tls.domain` Example" | ||||
|  | ||||
|     * A certificate for the domains `example.com` (main) and `*.example.org` (SAN) | ||||
|       is requested: | ||||
|        | ||||
|  | ||||
|     --8<-- "content/https/include-acme-multiple-domains-example.md" | ||||
|  | ||||
| ## Automatic Renewals | ||||
|  | ||||
| Traefik automatically tracks the expiry date of ACME certificates it generates. | ||||
|  | ||||
| If there are less than 30 days remaining before the certificate expires, Traefik will attempt to renew it automatically. | ||||
| By default, Traefik manages 90 days certificates, | ||||
| and starts to renew certificates 30 days before their expiry. | ||||
|  | ||||
| When using a certificate resolver that issues certificates with custom durations, | ||||
| one can configure the certificates' duration with the [`certificatesDuration`](#certificatesduration) option. | ||||
|  | ||||
| !!! info "" | ||||
|     Certificates that are no longer used may still be renewed, as Traefik does not currently check if the certificate is being used before renewing. | ||||
| @@ -154,7 +169,9 @@ When using LetsEncrypt with kubernetes, there are some known caveats with both t | ||||
|  | ||||
| ## The Different ACME Challenges | ||||
|  | ||||
| !!! important "Defining a certificates resolver does not result in all routers automatically using it. Each router that is supposed to use the resolver must [reference](../routing/routers/index.md#certresolver) it." | ||||
| !!! warning "Defining one ACME challenge is a requirement for a certificate resolver to be functional." | ||||
|  | ||||
| !!! important "Defining a certificate resolver does not result in all routers automatically using it. Each router that is supposed to use the resolver must [reference](../routing/routers/index.md#certresolver) it." | ||||
|  | ||||
| ### `tlsChallenge` | ||||
|  | ||||
| @@ -165,12 +182,6 @@ when using the `TLS-ALPN-01` challenge, Traefik must be reachable by Let's Encry | ||||
|  | ||||
| ??? example "Configuring the `tlsChallenge`" | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       # ... | ||||
|       [certificatesResolvers.myresolver.acme.tlsChallenge] | ||||
|     ``` | ||||
|  | ||||
|     ```yaml tab="File (YAML)" | ||||
|     certificatesResolvers: | ||||
|       myresolver: | ||||
| @@ -178,7 +189,13 @@ when using the `TLS-ALPN-01` challenge, Traefik must be reachable by Let's Encry | ||||
|           # ... | ||||
|           tlsChallenge: {} | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       # ... | ||||
|       [certificatesResolvers.myresolver.acme.tlsChallenge] | ||||
|     ``` | ||||
|  | ||||
|     ```bash tab="CLI" | ||||
|     # ... | ||||
|     --certificatesresolvers.myresolver.acme.tlschallenge=true | ||||
| @@ -193,28 +210,14 @@ when using the `HTTP-01` challenge, `certificatesresolvers.myresolver.acme.httpc | ||||
|  | ||||
| ??? example "Using an EntryPoint Called web for the `httpChallenge`" | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [entryPoints] | ||||
|       [entryPoints.web] | ||||
|         address = ":80" | ||||
|        | ||||
|       [entryPoints.websecure] | ||||
|         address = ":443" | ||||
|      | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       # ... | ||||
|       [certificatesResolvers.myresolver.acme.httpChallenge] | ||||
|         entryPoint = "web" | ||||
|     ``` | ||||
|  | ||||
|     ```yaml tab="File (YAML)" | ||||
|     entryPoints: | ||||
|       web: | ||||
|         address: ":80" | ||||
|      | ||||
|  | ||||
|       websecure: | ||||
|         address: ":443" | ||||
|      | ||||
|  | ||||
|     certificatesResolvers: | ||||
|       myresolver: | ||||
|         acme: | ||||
| @@ -222,7 +225,21 @@ when using the `HTTP-01` challenge, `certificatesresolvers.myresolver.acme.httpc | ||||
|           httpChallenge: | ||||
|             entryPoint: web | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [entryPoints] | ||||
|       [entryPoints.web] | ||||
|         address = ":80" | ||||
|  | ||||
|       [entryPoints.websecure] | ||||
|         address = ":443" | ||||
|  | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       # ... | ||||
|       [certificatesResolvers.myresolver.acme.httpChallenge] | ||||
|         entryPoint = "web" | ||||
|     ``` | ||||
|  | ||||
|     ```bash tab="CLI" | ||||
|     --entrypoints.web.address=:80 | ||||
|     --entrypoints.websecure.address=:443 | ||||
| @@ -239,15 +256,6 @@ Use the `DNS-01` challenge to generate and renew ACME certificates by provisioni | ||||
|  | ||||
| ??? example "Configuring a `dnsChallenge` with the DigitalOcean Provider" | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       # ... | ||||
|       [certificatesResolvers.myresolver.acme.dnsChallenge] | ||||
|         provider = "digitalocean" | ||||
|         delayBeforeCheck = 0 | ||||
|     # ... | ||||
|     ``` | ||||
|      | ||||
|     ```yaml tab="File (YAML)" | ||||
|     certificatesResolvers: | ||||
|       myresolver: | ||||
| @@ -258,7 +266,16 @@ Use the `DNS-01` challenge to generate and renew ACME certificates by provisioni | ||||
|             delayBeforeCheck: 0 | ||||
|         # ... | ||||
|     ``` | ||||
|      | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       # ... | ||||
|       [certificatesResolvers.myresolver.acme.dnsChallenge] | ||||
|         provider = "digitalocean" | ||||
|         delayBeforeCheck = 0 | ||||
|     # ... | ||||
|     ``` | ||||
|  | ||||
|     ```bash tab="CLI" | ||||
|     # ... | ||||
|     --certificatesresolvers.myresolver.acme.dnschallenge.provider=digitalocean | ||||
| @@ -266,11 +283,22 @@ Use the `DNS-01` challenge to generate and renew ACME certificates by provisioni | ||||
|     # ... | ||||
|     ``` | ||||
|  | ||||
|     !!! important | ||||
|         A `provider` is mandatory. | ||||
| !!! warning "`CNAME` support" | ||||
|  | ||||
|     `CNAME` are supported (and sometimes even [encouraged](https://letsencrypt.org/2019/10/09/onboarding-your-customers-with-lets-encrypt-and-acme.html#the-advantages-of-a-cname)), | ||||
|     but there are a few cases where they can be [problematic](../../getting-started/faq/#why-does-lets-encrypt-wildcard-certificate-renewalgeneration-with-dns-challenge-fail). | ||||
|  | ||||
|     If needed, `CNAME` support can be disabled with the following environment variable: | ||||
|  | ||||
|     ```bash | ||||
|     LEGO_DISABLE_CNAME_SUPPORT=true | ||||
|     ``` | ||||
|  | ||||
| !!! important | ||||
|     A `provider` is mandatory. | ||||
|  | ||||
| #### `providers` | ||||
|   | ||||
|  | ||||
| Here is a list of supported `providers`, that can automate the DNS verification, | ||||
| along with the required environment variables and their [wildcard & root domain support](#wildcard-domains). | ||||
| Do not hesitate to complete it. | ||||
| @@ -280,92 +308,141 @@ For example, `CF_API_EMAIL_FILE=/run/secrets/traefik_cf-api-email` could be used | ||||
|  | ||||
| For complete details, refer to your provider's _Additional configuration_ link. | ||||
|  | ||||
| | Provider Name                                               | Provider Code  | Environment Variables                                                                                                                       |                                                                             | | ||||
| |-------------------------------------------------------------|----------------|---------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------| | ||||
| | [ACME DNS](https://github.com/joohoi/acme-dns)              | `acme-dns`     | `ACME_DNS_API_BASE`, `ACME_DNS_STORAGE_PATH`                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/acme-dns)     | | ||||
| | [Alibaba Cloud](https://www.alibabacloud.com)               | `alidns`       | `ALICLOUD_ACCESS_KEY`, `ALICLOUD_SECRET_KEY`, `ALICLOUD_REGION_ID`                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/alidns)       | | ||||
| | [ArvanCloud](https://arvancloud.com)                        | `arvancloud`   | `ARVANCLOUD_API_KEY`                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/arvancloud)   | | ||||
| | [Auroradns](https://www.pcextreme.com/dns-health-checks)    | `auroradns`    | `AURORA_USER_ID`, `AURORA_KEY`, `AURORA_ENDPOINT`                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/auroradns)    | | ||||
| | [Autodns](https://www.internetx.com/domains/autodns/)       | `autodns`      | `AUTODNS_API_USER`, `AUTODNS_API_PASSWORD`                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/autodns)      | | ||||
| | [Azure](https://azure.microsoft.com/services/dns/)          | `azure`        | `AZURE_CLIENT_ID`, `AZURE_CLIENT_SECRET`, `AZURE_SUBSCRIPTION_ID`, `AZURE_TENANT_ID`, `AZURE_RESOURCE_GROUP`, `[AZURE_METADATA_ENDPOINT]`   | [Additional configuration](https://go-acme.github.io/lego/dns/azure)        | | ||||
| | [Bindman](https://github.com/labbsr0x/bindman-dns-webhook)  | `bindman`      | `BINDMAN_MANAGER_ADDRESS`                                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/bindman)      | | ||||
| | [Blue Cat](https://www.bluecatnetworks.com/)                | `bluecat`      | `BLUECAT_SERVER_URL`, `BLUECAT_USER_NAME`, `BLUECAT_PASSWORD`, `BLUECAT_CONFIG_NAME`, `BLUECAT_DNS_VIEW`                                    | [Additional configuration](https://go-acme.github.io/lego/dns/bluecat)      | | ||||
| | [Checkdomain](https://www.checkdomain.de/)                  | `checkdomain`  | `CHECKDOMAIN_TOKEN`,                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/checkdomain/) | | ||||
| | [CloudDNS](https://vshosting.eu/)                           | `clouddns`     | `CLOUDDNS_CLIENT_ID`, `CLOUDDNS_EMAIL`, `CLOUDDNS_PASSWORD`                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/clouddns)     | | ||||
| | [ClouDNS](https://www.cloudns.net/)                         | `cloudns`      | `CLOUDNS_AUTH_ID`, `CLOUDNS_AUTH_PASSWORD`                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/cloudns)      | | ||||
| | [Cloudflare](https://www.cloudflare.com)                    | `cloudflare`   | `CF_API_EMAIL`, `CF_API_KEY` [^5] or `CF_DNS_API_TOKEN`, `[CF_ZONE_API_TOKEN]`                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/cloudflare)   | | ||||
| | [CloudXNS](https://www.cloudxns.net)                        | `cloudxns`     | `CLOUDXNS_API_KEY`, `CLOUDXNS_SECRET_KEY`                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/cloudxns)     | | ||||
| | [ConoHa](https://www.conoha.jp)                             | `conoha`       | `CONOHA_TENANT_ID`, `CONOHA_API_USERNAME`, `CONOHA_API_PASSWORD`                                                                            | [Additional configuration](https://go-acme.github.io/lego/dns/conoha)       | | ||||
| | [Constellix](https://constellix.com)                        | `constellix`   | `CONSTELLIX_API_KEY`, `CONSTELLIX_SECRET_KEY`                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/constellix)   | | ||||
| | [deSEC](https://desec.io)                                   | `desec`        | `DESEC_TOKEN`                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/desec)        | | ||||
| | [DigitalOcean](https://www.digitalocean.com)                | `digitalocean` | `DO_AUTH_TOKEN`                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/digitalocean) | | ||||
| | [DNSimple](https://dnsimple.com)                            | `dnsimple`     | `DNSIMPLE_OAUTH_TOKEN`, `DNSIMPLE_BASE_URL`                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/dnsimple)     | | ||||
| | [DNS Made Easy](https://dnsmadeeasy.com)                    | `dnsmadeeasy`  | `DNSMADEEASY_API_KEY`, `DNSMADEEASY_API_SECRET`, `DNSMADEEASY_SANDBOX`                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/dnsmadeeasy)  | | ||||
| | [DNSPod](https://www.dnspod.com/)                           | `dnspod`       | `DNSPOD_API_KEY`                                                                                                                            | [Additional configuration](https://go-acme.github.io/lego/dns/dnspod)       | | ||||
| | [Domain Offensive (do.de)](https://www.do.de/)              | `dode`         | `DODE_TOKEN`                                                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/dode)         | | ||||
| | [DreamHost](https://www.dreamhost.com/)                     | `dreamhost`    | `DREAMHOST_API_KEY`                                                                                                                         | [Additional configuration](https://go-acme.github.io/lego/dns/dreamhost)    | | ||||
| | [Duck DNS](https://www.duckdns.org/)                        | `duckdns`      | `DUCKDNS_TOKEN`                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/duckdns)      | | ||||
| | [Dyn](https://dyn.com)                                      | `dyn`          | `DYN_CUSTOMER_NAME`, `DYN_USER_NAME`, `DYN_PASSWORD`                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/dyn)          | | ||||
| | [Dynu](https://www.dynu.com)                                | `dynu`         | `DYNU_API_KEY`                                                                                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/dynu)         | | ||||
| | [EasyDNS](https://easydns.com/)                             | `easydns`      | `EASYDNS_TOKEN`, `EASYDNS_KEY`                                                                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/easydns)      | | ||||
| | External Program                                            | `exec`         | `EXEC_PATH`                                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/exec)         | | ||||
| | [Exoscale](https://www.exoscale.com)                        | `exoscale`     | `EXOSCALE_API_KEY`, `EXOSCALE_API_SECRET`, `EXOSCALE_ENDPOINT`                                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/exoscale)     | | ||||
| | [Fast DNS](https://www.akamai.com/)                         | `fastdns`      | `AKAMAI_CLIENT_TOKEN`,  `AKAMAI_CLIENT_SECRET`,  `AKAMAI_ACCESS_TOKEN`                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/fastdns)      | | ||||
| | [Gandi](https://www.gandi.net)                              | `gandi`        | `GANDI_API_KEY`                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/gandi)        | | ||||
| | [Gandi v5](http://doc.livedns.gandi.net)                    | `gandiv5`      | `GANDIV5_API_KEY`                                                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/gandiv5)      | | ||||
| | [Glesys](https://glesys.com/)                               | `glesys`       | `GLESYS_API_USER`, `GLESYS_API_KEY`, `GLESYS_DOMAIN`                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/glesys)       | | ||||
| | [GoDaddy](https://godaddy.com/)                             | `godaddy`      | `GODADDY_API_KEY`, `GODADDY_API_SECRET`                                                                                                     | [Additional configuration](https://go-acme.github.io/lego/dns/godaddy)      | | ||||
| | [Google Cloud DNS](https://cloud.google.com/dns/docs/)      | `gcloud`       | `GCE_PROJECT`, Application Default Credentials [^2] [^3], [`GCE_SERVICE_ACCOUNT_FILE`]                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/gcloud)       | | ||||
| | [Hetzner](https://hetzner.com)                              | `hetzner`      | `HETZNER_API_KEY`                                                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/hetzner)      | | ||||
| | [hosting.de](https://www.hosting.de)                        | `hostingde`    | `HOSTINGDE_API_KEY`, `HOSTINGDE_ZONE_NAME`                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/hostingde)    | | ||||
| | HTTP request                                                | `httpreq`      | `HTTPREQ_ENDPOINT`, `HTTPREQ_MODE`, `HTTPREQ_USERNAME`, `HTTPREQ_PASSWORD` [^1]                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/httpreq)      | | ||||
| | [IIJ](https://www.iij.ad.jp/)                               | `iij`          | `IIJ_API_ACCESS_KEY`, `IIJ_API_SECRET_KEY`, `IIJ_DO_SERVICE_CODE`                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/iij)          | | ||||
| | [INWX](https://www.inwx.de/en)                              | `inwx`         | `INWX_USERNAME`, `INWX_PASSWORD`                                                                                                            | [Additional configuration](https://go-acme.github.io/lego/dns/inwx)         | | ||||
| | [Joker.com](https://joker.com)                              | `joker`        | `JOKER_API_KEY` or `JOKER_USERNAME`, `JOKER_PASSWORD`                                                                                       | [Additional configuration](https://go-acme.github.io/lego/dns/joker)        | | ||||
| | [Lightsail](https://aws.amazon.com/lightsail/)              | `lightsail`    | `AWS_ACCESS_KEY_ID`, `AWS_SECRET_ACCESS_KEY`, `DNS_ZONE`                                                                                    | [Additional configuration](https://go-acme.github.io/lego/dns/lightsail)    | | ||||
| | [Linode](https://www.linode.com)                            | `linode`       | `LINODE_API_KEY`                                                                                                                            | [Additional configuration](https://go-acme.github.io/lego/dns/linode)       | | ||||
| | [Linode v4](https://www.linode.com)                         | `linodev4`     | `LINODE_TOKEN`                                                                                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/linodev4)     | | ||||
| | [Liquid Web](https://www.liquidweb.com/)                    | `liquidweb`    | `LIQUID_WEB_PASSWORD`, `LIQUID_WEB_USERNAME`, `LIQUID_WEB_ZONE`                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/liquidweb)    | | ||||
| | [LuaDNS](https://luadns.com)                                | `luadns`       | `LUADNS_API_USERNAME`, `LUADNS_API_TOKEN`                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/luadns)       | | ||||
| | manual                                                      | `manual`       | none, but you need to run Traefik interactively [^4], turn on debug log to see instructions and press <kbd>Enter</kbd>.                     |                                                                             | | ||||
| | [MyDNS.jp](https://www.mydns.jp/)                           | `mydnsjp`      | `MYDNSJP_MASTER_ID`, `MYDNSJP_PASSWORD`                                                                                                     | [Additional configuration](https://go-acme.github.io/lego/dns/mydnsjp)      | | ||||
| | [Mythic Beasts](https://www.mythic-beasts.com)              | `mythicbeasts` | `MYTHICBEASTS_USER_NAME`, `MYTHICBEASTS_PASSWORD`                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/mythicbeasts) | | ||||
| | [Namecheap](https://www.namecheap.com)                      | `namecheap`    | `NAMECHEAP_API_USER`, `NAMECHEAP_API_KEY`                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/namecheap)    | | ||||
| | [name.com](https://www.name.com/)                           | `namedotcom`   | `NAMECOM_USERNAME`, `NAMECOM_API_TOKEN`, `NAMECOM_SERVER`                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/namedotcom)   | | ||||
| | [Namesilo](https://www.namesilo.com/)                       | `namesilo`     | `NAMESILO_API_KEY`                                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/namesilo)     | | ||||
| | [Netcup](https://www.netcup.eu/)                            | `netcup`       | `NETCUP_CUSTOMER_NUMBER`, `NETCUP_API_KEY`, `NETCUP_API_PASSWORD`                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/netcup)       | | ||||
| | [Netlify](https://www.netlify.com)                          | `netlify`      | `NETLIFY_TOKEN`                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/netlify)      | | ||||
| | [NIFCloud](https://cloud.nifty.com/service/dns.htm)         | `nifcloud`     | `NIFCLOUD_ACCESS_KEY_ID`, `NIFCLOUD_SECRET_ACCESS_KEY`                                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/nifcloud)     | | ||||
| | [Ns1](https://ns1.com/)                                     | `ns1`          | `NS1_API_KEY`                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/ns1)          | | ||||
| | [Open Telekom Cloud](https://cloud.telekom.de)              | `otc`          | `OTC_DOMAIN_NAME`, `OTC_USER_NAME`, `OTC_PASSWORD`, `OTC_PROJECT_NAME`, `OTC_IDENTITY_ENDPOINT`                                             | [Additional configuration](https://go-acme.github.io/lego/dns/otc)          | | ||||
| | [OVH](https://www.ovh.com)                                  | `ovh`          | `OVH_ENDPOINT`, `OVH_APPLICATION_KEY`, `OVH_APPLICATION_SECRET`, `OVH_CONSUMER_KEY`                                                         | [Additional configuration](https://go-acme.github.io/lego/dns/ovh)          | | ||||
| | [Openstack Designate](https://docs.openstack.org/designate) | `designate`    | `OS_AUTH_URL`, `OS_USERNAME`, `OS_PASSWORD`, `OS_TENANT_NAME`, `OS_REGION_NAME`                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/designate)    | | ||||
| | [Oracle Cloud](https://cloud.oracle.com/home)               | `oraclecloud`  | `OCI_COMPARTMENT_OCID`, `OCI_PRIVKEY_FILE`, `OCI_PRIVKEY_PASS`, `OCI_PUBKEY_FINGERPRINT`, `OCI_REGION`, `OCI_TENANCY_OCID`, `OCI_USER_OCID` | [Additional configuration](https://go-acme.github.io/lego/dns/oraclecloud)  | | ||||
| | [PowerDNS](https://www.powerdns.com)                        | `pdns`         | `PDNS_API_KEY`, `PDNS_API_URL`                                                                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/pdns)         | | ||||
| | [Rackspace](https://www.rackspace.com/cloud/dns)            | `rackspace`    | `RACKSPACE_USER`, `RACKSPACE_API_KEY`                                                                                                       | [Additional configuration](https://go-acme.github.io/lego/dns/rackspace)    | | ||||
| | [reg.ru](https://www.reg.ru)                                | `regru`        | `REGRU_USERNAME`, `REGRU_PASSWORD`                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/regru)        | | ||||
| | [RFC2136](https://tools.ietf.org/html/rfc2136)              | `rfc2136`      | `RFC2136_TSIG_KEY`, `RFC2136_TSIG_SECRET`, `RFC2136_TSIG_ALGORITHM`, `RFC2136_NAMESERVER`                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/rfc2136)      | | ||||
| | [Route 53](https://aws.amazon.com/route53/)                 | `route53`      | `AWS_ACCESS_KEY_ID`, `AWS_SECRET_ACCESS_KEY`, `[AWS_REGION]`, `[AWS_HOSTED_ZONE_ID]` or a configured user/instance IAM profile.             | [Additional configuration](https://go-acme.github.io/lego/dns/route53)      | | ||||
| | [RimuHosting](https://rimuhosting.com)                      | `rimuhosting`  | `RIMUHOSTING_API_KEY`                                                                                                                       | [Additional configuration](https://go-acme.github.io/lego/dns/rimuhosting)  | | ||||
| | [Sakura Cloud](https://cloud.sakura.ad.jp/)                 | `sakuracloud`  | `SAKURACLOUD_ACCESS_TOKEN`, `SAKURACLOUD_ACCESS_TOKEN_SECRET`                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/sakuracloud)  | | ||||
| | [Scaleway](https://www.scaleway.com)                        | `scaleway`     | `SCALEWAY_API_TOKEN`                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/scaleway)     | | ||||
| | [Selectel](https://selectel.ru/en/)                         | `selectel`     | `SELECTEL_API_TOKEN`                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/selectel)     | | ||||
| | [Servercow](https://servercow.de)                           | `servercow`    | `SERVERCOW_USERNAME`, `SERVERCOW_PASSWORD`                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/servercow)    | | ||||
| | [Stackpath](https://www.stackpath.com/)                     | `stackpath`    | `STACKPATH_CLIENT_ID`, `STACKPATH_CLIENT_SECRET`, `STACKPATH_STACK_ID`                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/stackpath)    | | ||||
| | [TransIP](https://www.transip.nl/)                          | `transip`      | `TRANSIP_ACCOUNT_NAME`, `TRANSIP_PRIVATE_KEY_PATH`                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/transip)      | | ||||
| | [VegaDNS](https://github.com/shupp/VegaDNS-API)             | `vegadns`      | `SECRET_VEGADNS_KEY`, `SECRET_VEGADNS_SECRET`, `VEGADNS_URL`                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/vegadns)      | | ||||
| | [Versio](https://www.versio.nl/domeinnamen)                 | `versio`       | `VERSIO_USERNAME`, `VERSIO_PASSWORD`                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/versio)       | | ||||
| | [Vscale](https://vscale.io/)                                | `vscale`       | `VSCALE_API_TOKEN`                                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/vscale)       | | ||||
| | [VULTR](https://www.vultr.com)                              | `vultr`        | `VULTR_API_KEY`                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/vultr)        | | ||||
| | [Yandex](https://yandex.com)                                | `yandex`       | `YANDEX_PDD_TOKEN`                                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/yandex)       | | ||||
| | [Zone.ee](https://www.zone.ee)                              | `zoneee`       | `ZONEEE_API_USER`, `ZONEEE_API_KEY`                                                                                                         | [Additional configuration](https://go-acme.github.io/lego/dns/zoneee)       | | ||||
| | [Zonomi](https://zonomi.com)                                | `zonomi`       | `ZONOMI_API_KEY`                                                                                                                            | [Additional configuration](https://go-acme.github.io/lego/dns/zonomi)       | | ||||
| | Provider Name                                                          | Provider Code      | Environment Variables                                                                                                                                                            |                                                                                 | | ||||
| |------------------------------------------------------------------------|--------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------------| | ||||
| | [ACME DNS](https://github.com/joohoi/acme-dns)                         | `acme-dns`         | `ACME_DNS_API_BASE`, `ACME_DNS_STORAGE_PATH`                                                                                                                                     | [Additional configuration](https://go-acme.github.io/lego/dns/acme-dns)         | | ||||
| | [Alibaba Cloud](https://www.alibabacloud.com)                          | `alidns`           | `ALICLOUD_ACCESS_KEY`, `ALICLOUD_SECRET_KEY`, `ALICLOUD_REGION_ID`                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/alidns)           | | ||||
| | [all-inkl](https://all-inkl.com)                                       | `allinkl`          | `ALL_INKL_LOGIN`, `ALL_INKL_PASSWORD`                                                                                                                                            | [Additional configuration](https://go-acme.github.io/lego/dns/allinkl)          | | ||||
| | [ArvanCloud](https://www.arvancloud.ir/en)                            | `arvancloud`       | `ARVANCLOUD_API_KEY`                                                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/arvancloud)       | | ||||
| | [Auroradns](https://www.pcextreme.com/dns-health-checks)               | `auroradns`        | `AURORA_USER_ID`, `AURORA_KEY`, `AURORA_ENDPOINT`                                                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/auroradns)        | | ||||
| | [Autodns](https://www.internetx.com/domains/autodns/)                  | `autodns`          | `AUTODNS_API_USER`, `AUTODNS_API_PASSWORD`                                                                                                                                       | [Additional configuration](https://go-acme.github.io/lego/dns/autodns)          | | ||||
| | [Azure](https://azure.microsoft.com/services/dns/)  (DEPRECATED)       | `azure`            | `AZURE_CLIENT_ID`, `AZURE_CLIENT_SECRET`, `AZURE_SUBSCRIPTION_ID`, `AZURE_TENANT_ID`, `AZURE_RESOURCE_GROUP`, `[AZURE_METADATA_ENDPOINT]`                                        | [Additional configuration](https://go-acme.github.io/lego/dns/azure)            | | ||||
| | [AzureDNS](https://azure.microsoft.com/services/dns/)                  | `azuredns`         | `AZURE_CLIENT_ID`, `AZURE_CLIENT_SECRET`, `AZURE_TENANT_ID`, `AZURE_SUBSCRIPTION_ID`, `AZURE_RESOURCE_GROUP`, `[AZURE_ENVIRONMENT]`, `[AZURE_PRIVATE_ZONE]`, `[AZURE_ZONE_NAME]` | [Additional configuration](https://go-acme.github.io/lego/dns/azuredns)         | | ||||
| | [Bindman](https://github.com/labbsr0x/bindman-dns-webhook)             | `bindman`          | `BINDMAN_MANAGER_ADDRESS`                                                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/bindman)          | | ||||
| | [Blue Cat](https://www.bluecatnetworks.com/)                           | `bluecat`          | `BLUECAT_SERVER_URL`, `BLUECAT_USER_NAME`, `BLUECAT_PASSWORD`, `BLUECAT_CONFIG_NAME`, `BLUECAT_DNS_VIEW`                                                                         | [Additional configuration](https://go-acme.github.io/lego/dns/bluecat)          | | ||||
| | [Brandit](https://www.brandit.com)                                     | `brandit`          | `BRANDIT_API_USERNAME`, `BRANDIT_API_KEY`                                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/brandit)          | | ||||
| | [Bunny](https://bunny.net)                                             | `bunny`            | `BUNNY_API_KEY`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/bunny)            | | ||||
| | [Checkdomain](https://www.checkdomain.de/)                             | `checkdomain`      | `CHECKDOMAIN_TOKEN`,                                                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/checkdomain/)     | | ||||
| | [Civo](https://www.civo.com/)                                          | `civo`             | `CIVO_TOKEN`                                                                                                                                                                     | [Additional configuration](https://go-acme.github.io/lego/dns/civo)             | | ||||
| | [Cloud.ru](https://cloud.ru)                                           | `cloudru`          | `CLOUDRU_SERVICE_INSTANCE_ID`, `CLOUDRU_KEY_ID`, `CLOUDRU_SECRET`                                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/cloudru)          | | ||||
| | [CloudDNS](https://vshosting.eu/)                                      | `clouddns`         | `CLOUDDNS_CLIENT_ID`, `CLOUDDNS_EMAIL`, `CLOUDDNS_PASSWORD`                                                                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/clouddns)         | | ||||
| | [Cloudflare](https://www.cloudflare.com)                               | `cloudflare`       | `CF_API_EMAIL`, `CF_API_KEY` [^5] or `CF_DNS_API_TOKEN`, `[CF_ZONE_API_TOKEN]`                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/cloudflare)       | | ||||
| | [ClouDNS](https://www.cloudns.net/)                                    | `cloudns`          | `CLOUDNS_AUTH_ID`, `CLOUDNS_AUTH_PASSWORD`                                                                                                                                       | [Additional configuration](https://go-acme.github.io/lego/dns/cloudns)          | | ||||
| | [CloudXNS](https://www.cloudxns.net)                                   | `cloudxns`         | `CLOUDXNS_API_KEY`, `CLOUDXNS_SECRET_KEY`                                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/cloudxns)         | | ||||
| | [ConoHa](https://www.conoha.jp)                                        | `conoha`           | `CONOHA_TENANT_ID`, `CONOHA_API_USERNAME`, `CONOHA_API_PASSWORD`                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/conoha)           | | ||||
| | [Constellix](https://constellix.com)                                   | `constellix`       | `CONSTELLIX_API_KEY`, `CONSTELLIX_SECRET_KEY`                                                                                                                                    | [Additional configuration](https://go-acme.github.io/lego/dns/constellix)       | | ||||
| | [Derak Cloud](https://derak.cloud/)                                    | `derak`            | `DERAK_API_KEY`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/derak)            | | ||||
| | [deSEC](https://desec.io)                                              | `desec`            | `DESEC_TOKEN`                                                                                                                                                                    | [Additional configuration](https://go-acme.github.io/lego/dns/desec)            | | ||||
| | [DigitalOcean](https://www.digitalocean.com)                           | `digitalocean`     | `DO_AUTH_TOKEN`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/digitalocean)     | | ||||
| | [DNS Made Easy](https://dnsmadeeasy.com)                               | `dnsmadeeasy`      | `DNSMADEEASY_API_KEY`, `DNSMADEEASY_API_SECRET`, `DNSMADEEASY_SANDBOX`                                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/dnsmadeeasy)      | | ||||
| | [dnsHome.de](https://www.dnshome.de)                                   | `dnsHomede`        | `DNSHOMEDE_CREDENTIALS`                                                                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/dnshomede)        | | ||||
| | [DNSimple](https://dnsimple.com)                                       | `dnsimple`         | `DNSIMPLE_OAUTH_TOKEN`, `DNSIMPLE_BASE_URL`                                                                                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/dnsimple)         | | ||||
| | [DNSPod](https://www.dnspod.com/)                                      | `dnspod`           | `DNSPOD_API_KEY`                                                                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/dnspod)           | | ||||
| | [Domain Offensive (do.de)](https://www.do.de/)                         | `dode`             | `DODE_TOKEN`                                                                                                                                                                     | [Additional configuration](https://go-acme.github.io/lego/dns/dode)             | | ||||
| | [Domeneshop](https://domene.shop)                                      | `domeneshop`       | `DOMENESHOP_API_TOKEN`, `DOMENESHOP_API_SECRET`                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/domeneshop)       | | ||||
| | [DreamHost](https://www.dreamhost.com/)                                | `dreamhost`        | `DREAMHOST_API_KEY`                                                                                                                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/dreamhost)        | | ||||
| | [Duck DNS](https://www.duckdns.org/)                                   | `duckdns`          | `DUCKDNS_TOKEN`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/duckdns)          | | ||||
| | [Dyn](https://dyn.com)                                                 | `dyn`              | `DYN_CUSTOMER_NAME`, `DYN_USER_NAME`, `DYN_PASSWORD`                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/dyn)              | | ||||
| | [Dynu](https://www.dynu.com)                                           | `dynu`             | `DYNU_API_KEY`                                                                                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/dynu)             | | ||||
| | [EasyDNS](https://easydns.com/)                                        | `easydns`          | `EASYDNS_TOKEN`, `EASYDNS_KEY`                                                                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/easydns)          | | ||||
| | [EdgeDNS](https://www.akamai.com/)                                     | `edgedns`          | `AKAMAI_CLIENT_TOKEN`,  `AKAMAI_CLIENT_SECRET`,  `AKAMAI_ACCESS_TOKEN`                                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/edgedns)          | | ||||
| | [Efficient IP](https://efficientip.com)                                | `efficientip`      | `EFFICIENTIP_USERNAME`, `EFFICIENTIP_PASSWORD`, `EFFICIENTIP_HOSTNAME`, `EFFICIENTIP_DNS_NAME`                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/efficientip)      | | ||||
| | [Epik](https://www.epik.com)                                           | `epik`             | `EPIK_SIGNATURE`                                                                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/epik)             | | ||||
| | [Exoscale](https://www.exoscale.com)                                   | `exoscale`         | `EXOSCALE_API_KEY`, `EXOSCALE_API_SECRET`, `EXOSCALE_ENDPOINT`                                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/exoscale)         | | ||||
| | [Fast DNS](https://www.akamai.com/)                                    | `fastdns`          | `AKAMAI_CLIENT_TOKEN`,  `AKAMAI_CLIENT_SECRET`,  `AKAMAI_ACCESS_TOKEN`                                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/edgedns)          | | ||||
| | [Freemyip.com](https://freemyip.com)                                   | `freemyip`         | `FREEMYIP_TOKEN`                                                                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/freemyip)         | | ||||
| | [G-Core](https://gcore.com/dns/)                                       | `gcore`            | `GCORE_PERMANENT_API_TOKEN`                                                                                                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/gcore)            | | ||||
| | [Gandi v5](https://doc.livedns.gandi.net)                              | `gandiv5`          | `GANDIV5_API_KEY`                                                                                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/gandiv5)          | | ||||
| | [Gandi](https://www.gandi.net)                                         | `gandi`            | `GANDI_API_KEY`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/gandi)            | | ||||
| | [Glesys](https://glesys.com/)                                          | `glesys`           | `GLESYS_API_USER`, `GLESYS_API_KEY`, `GLESYS_DOMAIN`                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/glesys)           | | ||||
| | [GoDaddy](https://www.godaddy.com)                                     | `godaddy`          | `GODADDY_API_KEY`, `GODADDY_API_SECRET`                                                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/godaddy)          | | ||||
| | [Google Cloud DNS](https://cloud.google.com/dns/docs/)                 | `gcloud`           | `GCE_PROJECT`, Application Default Credentials [^2] [^3], [`GCE_SERVICE_ACCOUNT_FILE`]                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/gcloud)           | | ||||
| | [Google Domains](https://domains.google)                               | `googledomains`    | `GOOGLE_DOMAINS_ACCESS_TOKEN`                                                                                                                                                    | [Additional configuration](https://go-acme.github.io/lego/dns/googledomains)    | | ||||
| | [Hetzner](https://hetzner.com)                                         | `hetzner`          | `HETZNER_API_KEY`                                                                                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/hetzner)          | | ||||
| | [hosting.de](https://www.hosting.de)                                   | `hostingde`        | `HOSTINGDE_API_KEY`, `HOSTINGDE_ZONE_NAME`                                                                                                                                       | [Additional configuration](https://go-acme.github.io/lego/dns/hostingde)        | | ||||
| | [Hosttech](https://www.hosttech.eu)                                    | `hosttech`         | `HOSTTECH_API_KEY`                                                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/hosttech)         | | ||||
| | [Hurricane Electric](https://dns.he.net)                               | `hurricane`        | `HURRICANE_TOKENS` [^6]                                                                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/hurricane)        | | ||||
| | [HyperOne](https://www.hyperone.com)                                   | `hyperone`         | `HYPERONE_PASSPORT_LOCATION`, `HYPERONE_LOCATION_ID`                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/hyperone)         | | ||||
| | [IBM Cloud (SoftLayer)](https://www.ibm.com/cloud/)                    | `ibmcloud`         | `SOFTLAYER_USERNAME`, `SOFTLAYER_API_KEY`                                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/ibmcloud)         | | ||||
| | [IIJ DNS Platform Service](https://www.iij.ad.jp)                      | `iijdpf`           | `IIJ_DPF_API_TOKEN` , `IIJ_DPF_DPM_SERVICE_CODE`                                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/iijdpf)           | | ||||
| | [IIJ](https://www.iij.ad.jp/)                                          | `iij`              | `IIJ_API_ACCESS_KEY`, `IIJ_API_SECRET_KEY`, `IIJ_DO_SERVICE_CODE`                                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/iij)              | | ||||
| | [Infoblox](https://www.infoblox.com/)                                  | `infoblox`         | `INFOBLOX_USERNAME`, `INFOBLOX_PASSWORD`, `INFOBLOX_HOST`                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/infoblox)         | | ||||
| | [Infomaniak](https://www.infomaniak.com)                               | `infomaniak`       | `INFOMANIAK_ACCESS_TOKEN`                                                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/infomaniak)       | | ||||
| | [Internet.bs](https://internetbs.net)                                  | `internetbs`       | `INTERNET_BS_API_KEY`, `INTERNET_BS_PASSWORD`                                                                                                                                    | [Additional configuration](https://go-acme.github.io/lego/dns/internetbs)       | | ||||
| | [INWX](https://www.inwx.de/en)                                         | `inwx`             | `INWX_USERNAME`, `INWX_PASSWORD`                                                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/inwx)             | | ||||
| | [ionos](https://ionos.com/)                                            | `ionos`            | `IONOS_API_KEY`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/ionos)            | | ||||
| | [IPv64](https://ipv64.net)                                             | `ipv64`            | `IPV64_API_KEY`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/ipv64)            | | ||||
| | [iwantmyname](https://iwantmyname.com)                                 | `iwantmyname`      | `IWANTMYNAME_USERNAME` , `IWANTMYNAME_PASSWORD`                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/iwantmyname)      | | ||||
| | [Joker.com](https://joker.com)                                         | `joker`            | `JOKER_API_MODE` with `JOKER_API_KEY` or `JOKER_USERNAME`, `JOKER_PASSWORD`                                                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/joker)            | | ||||
| | [Liara](https://liara.ir)                                              | `liara`            | `LIARA_API_KEY`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/liara)            | | ||||
| | [Lightsail](https://aws.amazon.com/lightsail/)                         | `lightsail`        | `AWS_ACCESS_KEY_ID`, `AWS_SECRET_ACCESS_KEY`, `DNS_ZONE`                                                                                                                         | [Additional configuration](https://go-acme.github.io/lego/dns/lightsail)        | | ||||
| | [Linode v4](https://www.linode.com)                                    | `linode`           | `LINODE_TOKEN`                                                                                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/linode)           | | ||||
| | [Liquid Web](https://www.liquidweb.com/)                               | `liquidweb`        | `LIQUID_WEB_PASSWORD`, `LIQUID_WEB_USERNAME`, `LIQUID_WEB_ZONE`                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/liquidweb)        | | ||||
| | [Loopia](https://loopia.com/)                                          | `loopia`           | `LOOPIA_API_PASSWORD`, `LOOPIA_API_USER`                                                                                                                                         | [Additional configuration](https://go-acme.github.io/lego/dns/loopia)           | | ||||
| | [LuaDNS](https://luadns.com)                                           | `luadns`           | `LUADNS_API_USERNAME`, `LUADNS_API_TOKEN`                                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/luadns)           | | ||||
| | [Metaname](https://metaname.net)                                       | `metaname`         | `METANAME_ACCOUNT_REFERENCE`, `METANAME_API_KEY`                                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/metaname)         | | ||||
| | [MyDNS.jp](https://www.mydns.jp/)                                      | `mydnsjp`          | `MYDNSJP_MASTER_ID`, `MYDNSJP_PASSWORD`                                                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/mydnsjp)          | | ||||
| | [Mythic Beasts](https://www.mythic-beasts.com)                         | `mythicbeasts`     | `MYTHICBEASTS_USER_NAME`, `MYTHICBEASTS_PASSWORD`                                                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/mythicbeasts)     | | ||||
| | [name.com](https://www.name.com/)                                      | `namedotcom`       | `NAMECOM_USERNAME`, `NAMECOM_API_TOKEN`, `NAMECOM_SERVER`                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/namedotcom)       | | ||||
| | [Namecheap](https://www.namecheap.com)                                 | `namecheap`        | `NAMECHEAP_API_USER`, `NAMECHEAP_API_KEY`                                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/namecheap)        | | ||||
| | [Namesilo](https://www.namesilo.com/)                                  | `namesilo`         | `NAMESILO_API_KEY`                                                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/namesilo)         | | ||||
| | [NearlyFreeSpeech.NET](https://www.nearlyfreespeech.net/)              | `nearlyfreespeech` | `NEARLYFREESPEECH_API_KEY`, `NEARLYFREESPEECH_LOGIN`                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/nearlyfreespeech) | | ||||
| | [Netcup](https://www.netcup.eu/)                                       | `netcup`           | `NETCUP_CUSTOMER_NUMBER`, `NETCUP_API_KEY`, `NETCUP_API_PASSWORD`                                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/netcup)           | | ||||
| | [Netlify](https://www.netlify.com)                                     | `netlify`          | `NETLIFY_TOKEN`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/netlify)          | | ||||
| | [Nicmanager](https://www.nicmanager.com)                               | `nicmanager`       | `NICMANAGER_API_EMAIL`, `NICMANAGER_API_PASSWORD`                                                                                                                                | [Additional configuration](https://go-acme.github.io/lego/dns/nicmanager)       | | ||||
| | [NIFCloud](https://cloud.nifty.com/service/dns.htm)                    | `nifcloud`         | `NIFCLOUD_ACCESS_KEY_ID`, `NIFCLOUD_SECRET_ACCESS_KEY`                                                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/nifcloud)         | | ||||
| | [Njalla](https://njal.la)                                              | `njalla`           | `NJALLA_TOKEN`                                                                                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/njalla)           | | ||||
| | [Nodion](https://www.nodion.com)                                       | `nodion`           | `NODION_API_TOKEN`                                                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/nodion)           | | ||||
| | [NS1](https://ns1.com/)                                                | `ns1`              | `NS1_API_KEY`                                                                                                                                                                    | [Additional configuration](https://go-acme.github.io/lego/dns/ns1)              | | ||||
| | [Open Telekom Cloud](https://cloud.telekom.de)                         | `otc`              | `OTC_DOMAIN_NAME`, `OTC_USER_NAME`, `OTC_PASSWORD`, `OTC_PROJECT_NAME`, `OTC_IDENTITY_ENDPOINT`                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/otc)              | | ||||
| | [Openstack Designate](https://docs.openstack.org/designate)            | `designate`        | `OS_AUTH_URL`, `OS_USERNAME`, `OS_PASSWORD`, `OS_TENANT_NAME`, `OS_REGION_NAME`                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/designate)        | | ||||
| | [Oracle Cloud](https://cloud.oracle.com/home)                          | `oraclecloud`      | `OCI_COMPARTMENT_OCID`, `OCI_PRIVKEY_FILE`, `OCI_PRIVKEY_PASS`, `OCI_PUBKEY_FINGERPRINT`, `OCI_REGION`, `OCI_TENANCY_OCID`, `OCI_USER_OCID`                                      | [Additional configuration](https://go-acme.github.io/lego/dns/oraclecloud)      | | ||||
| | [OVH](https://www.ovh.com)                                             | `ovh`              | `OVH_ENDPOINT`, `OVH_APPLICATION_KEY`, `OVH_APPLICATION_SECRET`, `OVH_CONSUMER_KEY`                                                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/ovh)              | | ||||
| | [Plesk](https://www.plesk.com)                                         | `plesk`            | `PLESK_SERVER_BASE_URL`, `PLESK_USERNAME`, `PLESK_PASSWORD`                                                                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/plesk)            | | ||||
| | [Porkbun](https://porkbun.com/)                                        | `porkbun`          | `PORKBUN_SECRET_API_KEY`, `PORKBUN_API_KEY`                                                                                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/porkbun)          | | ||||
| | [PowerDNS](https://www.powerdns.com)                                   | `pdns`             | `PDNS_API_KEY`, `PDNS_API_URL`                                                                                                                                                   | [Additional configuration](https://go-acme.github.io/lego/dns/pdns)             | | ||||
| | [Rackspace](https://www.rackspace.com/cloud/dns)                       | `rackspace`        | `RACKSPACE_USER`, `RACKSPACE_API_KEY`                                                                                                                                            | [Additional configuration](https://go-acme.github.io/lego/dns/rackspace)        | | ||||
| | [RcodeZero](https://www.rcodezero.at)                                  | `rcodezero`        | `RCODEZERO_API_TOKEN`                                                                                                                                                            | [Additional configuration](https://go-acme.github.io/lego/dns/rcodezero)        | | ||||
| | [reg.ru](https://www.reg.ru)                                           | `regru`            | `REGRU_USERNAME`, `REGRU_PASSWORD`                                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/regru)            | | ||||
| | [RFC2136](https://tools.ietf.org/html/rfc2136)                         | `rfc2136`          | `RFC2136_TSIG_KEY`, `RFC2136_TSIG_SECRET`, `RFC2136_TSIG_ALGORITHM`, `RFC2136_NAMESERVER`                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/rfc2136)          | | ||||
| | [RimuHosting](https://rimuhosting.com)                                 | `rimuhosting`      | `RIMUHOSTING_API_KEY`                                                                                                                                                            | [Additional configuration](https://go-acme.github.io/lego/dns/rimuhosting)      | | ||||
| | [Route 53](https://aws.amazon.com/route53/)                            | `route53`          | `AWS_ACCESS_KEY_ID`, `AWS_SECRET_ACCESS_KEY`, `[AWS_REGION]`, `[AWS_HOSTED_ZONE_ID]` or a configured user/instance IAM profile.                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/route53)          | | ||||
| | [Sakura Cloud](https://cloud.sakura.ad.jp/)                            | `sakuracloud`      | `SAKURACLOUD_ACCESS_TOKEN`, `SAKURACLOUD_ACCESS_TOKEN_SECRET`                                                                                                                    | [Additional configuration](https://go-acme.github.io/lego/dns/sakuracloud)      | | ||||
| | [Scaleway](https://www.scaleway.com)                                   | `scaleway`         | `SCALEWAY_API_TOKEN`                                                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/scaleway)         | | ||||
| | [Selectel](https://selectel.ru/en/)                                    | `selectel`         | `SELECTEL_API_TOKEN`                                                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/selectel)         | | ||||
| | [Servercow](https://servercow.de)                                      | `servercow`        | `SERVERCOW_USERNAME`, `SERVERCOW_PASSWORD`                                                                                                                                       | [Additional configuration](https://go-acme.github.io/lego/dns/servercow)        | | ||||
| | [Simply.com](https://www.simply.com/en/domains/)                       | `simply`           | `SIMPLY_ACCOUNT_NAME`, `SIMPLY_API_KEY`                                                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/simply)           | | ||||
| | [Sonic](https://www.sonic.com/)                                        | `sonic`            | `SONIC_USER_ID`, `SONIC_API_KEY`                                                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/sonic)            | | ||||
| | [Stackpath](https://www.stackpath.com/)                                | `stackpath`        | `STACKPATH_CLIENT_ID`, `STACKPATH_CLIENT_SECRET`, `STACKPATH_STACK_ID`                                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/stackpath)        | | ||||
| | [Tencent Cloud DNS](https://cloud.tencent.com/product/cns)             | `tencentcloud`     | `TENCENTCLOUD_SECRET_ID`, `TENCENTCLOUD_SECRET_KEY`                                                                                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/tencentcloud)     | | ||||
| | [TransIP](https://www.transip.nl/)                                     | `transip`          | `TRANSIP_ACCOUNT_NAME`, `TRANSIP_PRIVATE_KEY_PATH`                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/transip)          | | ||||
| | [UKFast SafeDNS](https://docs.ukfast.co.uk/domains/safedns/index.html) | `safedns`          | `SAFEDNS_AUTH_TOKEN`                                                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/safedns)          | | ||||
| | [Ultradns](https://neustarsecurityservices.com/dns-services)           | `ultradns`         | `ULTRADNS_USERNAME`, `ULTRADNS_PASSWORD`                                                                                                                                         | [Additional configuration](https://go-acme.github.io/lego/dns/ultradns)         | | ||||
| | [Variomedia](https://www.variomedia.de/)                               | `variomedia`       | `VARIOMEDIA_API_TOKEN`                                                                                                                                                           | [Additional configuration](https://go-acme.github.io/lego/dns/variomedia)       | | ||||
| | [VegaDNS](https://github.com/shupp/VegaDNS-API)                        | `vegadns`          | `SECRET_VEGADNS_KEY`, `SECRET_VEGADNS_SECRET`, `VEGADNS_URL`                                                                                                                     | [Additional configuration](https://go-acme.github.io/lego/dns/vegadns)          | | ||||
| | [Vercel](https://vercel.com)                                           | `vercel`           | `VERCEL_API_TOKEN`                                                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/vercel)           | | ||||
| | [Versio](https://www.versio.nl/domeinnamen)                            | `versio`           | `VERSIO_USERNAME`, `VERSIO_PASSWORD`                                                                                                                                             | [Additional configuration](https://go-acme.github.io/lego/dns/versio)           | | ||||
| | [VinylDNS](https://www.vinyldns.io)                                    | `vinyldns`         | `VINYLDNS_ACCESS_KEY`, `VINYLDNS_SECRET_KEY`, `VINYLDNS_HOST`                                                                                                                    | [Additional configuration](https://go-acme.github.io/lego/dns/vinyldns)         | | ||||
| | [VK Cloud](https://mcs.mail.ru/)                                       | `vkcloud`          | `VK_CLOUD_PASSWORD`, `VK_CLOUD_PROJECT_ID`, `VK_CLOUD_USERNAME`                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/vkcloud)          | | ||||
| | [Vscale](https://vscale.io/)                                           | `vscale`           | `VSCALE_API_TOKEN`                                                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/vscale)           | | ||||
| | [VULTR](https://www.vultr.com)                                         | `vultr`            | `VULTR_API_KEY`                                                                                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/vultr)            | | ||||
| | [Websupport](https://websupport.sk)                                    | `websupport`       | `WEBSUPPORT_API_KEY`, `WEBSUPPORT_SECRET`                                                                                                                                        | [Additional configuration](https://go-acme.github.io/lego/dns/websupport)       | | ||||
| | [WEDOS](https://www.wedos.com)                                         | `wedos`            | `WEDOS_USERNAME`, `WEDOS_WAPI_PASSWORD`                                                                                                                                          | [Additional configuration](https://go-acme.github.io/lego/dns/wedos)            | | ||||
| | [Yandex 360](https://360.yandex.ru)                                    | `yandex360`        | `YANDEX360_OAUTH_TOKEN`, `YANDEX360_ORG_ID`                                                                                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/yandex360)        | | ||||
| | [Yandex Cloud](https://cloud.yandex.com/en/)                           | `yandexcloud`      | `YANDEX_CLOUD_FOLDER_ID`, `YANDEX_CLOUD_IAM_TOKEN`                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/yandexcloud)      | | ||||
| | [Yandex](https://yandex.com)                                           | `yandex`           | `YANDEX_PDD_TOKEN`                                                                                                                                                               | [Additional configuration](https://go-acme.github.io/lego/dns/yandex)           | | ||||
| | [Zone.ee](https://www.zone.ee)                                         | `zoneee`           | `ZONEEE_API_USER`, `ZONEEE_API_KEY`                                                                                                                                              | [Additional configuration](https://go-acme.github.io/lego/dns/zoneee)           | | ||||
| | [Zonomi](https://zonomi.com)                                           | `zonomi`           | `ZONOMI_API_KEY`                                                                                                                                                                 | [Additional configuration](https://go-acme.github.io/lego/dns/zonomi)           | | ||||
| | External Program                                                       | `exec`             | `EXEC_PATH`                                                                                                                                                                      | [Additional configuration](https://go-acme.github.io/lego/dns/exec)             | | ||||
| | HTTP request                                                           | `httpreq`          | `HTTPREQ_ENDPOINT`, `HTTPREQ_MODE`, `HTTPREQ_USERNAME`, `HTTPREQ_PASSWORD` [^1]                                                                                                  | [Additional configuration](https://go-acme.github.io/lego/dns/httpreq)          | | ||||
| | manual                                                                 | `manual`           | none, but you need to run Traefik interactively [^4], turn on debug log to see instructions and press <kbd>Enter</kbd>.                                                          |                                                                                 | | ||||
|  | ||||
| [^1]: more information about the HTTP message format can be found [here](https://go-acme.github.io/lego/dns/httpreq/) | ||||
| [^2]: [providing_credentials_to_your_application](https://cloud.google.com/docs/authentication/production#providing_credentials_to_your_application) | ||||
| [^1]: More information about the HTTP message format can be found [here](https://go-acme.github.io/lego/dns/httpreq/). | ||||
| [^2]: [Providing credentials to your application](https://cloud.google.com/docs/authentication/production). | ||||
| [^3]: [google/default.go](https://github.com/golang/oauth2/blob/36a7019397c4c86cf59eeab3bc0d188bac444277/google/default.go#L61-L76) | ||||
| [^4]: `docker stack` remark: there is no way to support terminal attached to container when deploying with `docker stack`, so you might need to run container with `docker run -it` to generate certificates using `manual` provider. | ||||
| [^5]: The `Global API Key` needs to be used, not the `Origin CA Key`. | ||||
| [^6]: As explained in the [LEGO hurricane configuration](https://go-acme.github.io/lego/dns/hurricane/#credentials), each domain or wildcard (record name) needs a token. So each update of record name must be followed by an update of the `HURRICANE_TOKENS` variable, and a restart of Traefik. | ||||
|  | ||||
| !!! info "`delayBeforeCheck`" | ||||
|     By default, the `provider` verifies the TXT record _before_ letting ACME verify. | ||||
| @@ -376,14 +453,6 @@ For complete details, refer to your provider's _Additional configuration_ link. | ||||
|  | ||||
| Use custom DNS servers to resolve the FQDN authority. | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| [certificatesResolvers.myresolver.acme] | ||||
|   # ... | ||||
|   [certificatesResolvers.myresolver.acme.dnsChallenge] | ||||
|     # ... | ||||
|     resolvers = ["1.1.1.1:53", "8.8.8.8:53"] | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| certificatesResolvers: | ||||
|   myresolver: | ||||
| @@ -396,6 +465,14 @@ certificatesResolvers: | ||||
|           - "8.8.8.8:53" | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| [certificatesResolvers.myresolver.acme] | ||||
|   # ... | ||||
|   [certificatesResolvers.myresolver.acme.dnsChallenge] | ||||
|     # ... | ||||
|     resolvers = ["1.1.1.1:53", "8.8.8.8:53"] | ||||
| ``` | ||||
|  | ||||
| ```bash tab="CLI" | ||||
| # ... | ||||
| --certificatesresolvers.myresolver.acme.dnschallenge.resolvers=1.1.1.1:53,8.8.8.8:53 | ||||
| @@ -406,6 +483,35 @@ certificatesResolvers: | ||||
| [ACME V2](https://community.letsencrypt.org/t/acme-v2-and-wildcard-certificate-support-is-live/55579) supports wildcard certificates. | ||||
| As described in [Let's Encrypt's post](https://community.letsencrypt.org/t/staging-endpoint-for-acme-v2/49605) wildcard certificates can only be generated through a [`DNS-01` challenge](#dnschallenge). | ||||
|  | ||||
| ## External Account Binding | ||||
|  | ||||
| - `kid`: Key identifier from External CA | ||||
| - `hmacEncoded`: HMAC key from External CA, should be in Base64 URL Encoding without padding format | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| certificatesResolvers: | ||||
|   myresolver: | ||||
|     acme: | ||||
|       # ... | ||||
|       eab: | ||||
|         kid: abc-keyID-xyz | ||||
|         hmacEncoded: abc-hmac-xyz | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| [certificatesResolvers.myresolver.acme] | ||||
|   # ... | ||||
|   [certificatesResolvers.myresolver.acme.eab] | ||||
|     kid = "abc-keyID-xyz" | ||||
|     hmacEncoded = "abc-hmac-xyz" | ||||
| ``` | ||||
|  | ||||
| ```bash tab="CLI" | ||||
| # ... | ||||
| --certificatesresolvers.myresolver.acme.eab.kid=abc-keyID-xyz | ||||
| --certificatesresolvers.myresolver.acme.eab.hmacencoded=abc-hmac-xyz | ||||
| ``` | ||||
|  | ||||
| ## More Configuration | ||||
|  | ||||
| ### `caServer` | ||||
| @@ -419,13 +525,6 @@ The CA server to use: | ||||
|  | ||||
| ??? example "Using the Let's Encrypt staging server" | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       # ... | ||||
|       caServer = "https://acme-staging-v02.api.letsencrypt.org/directory" | ||||
|       # ... | ||||
|     ``` | ||||
|      | ||||
|     ```yaml tab="File (YAML)" | ||||
|     certificatesResolvers: | ||||
|       myresolver: | ||||
| @@ -435,6 +534,13 @@ The CA server to use: | ||||
|           # ... | ||||
|     ``` | ||||
|  | ||||
|     ```toml tab="File (TOML)" | ||||
|     [certificatesResolvers.myresolver.acme] | ||||
|       # ... | ||||
|       caServer = "https://acme-staging-v02.api.letsencrypt.org/directory" | ||||
|       # ... | ||||
|     ``` | ||||
|  | ||||
|     ```bash tab="CLI" | ||||
|     # ... | ||||
|     --certificatesresolvers.myresolver.acme.caserver=https://acme-staging-v02.api.letsencrypt.org/directory | ||||
| @@ -447,13 +553,6 @@ _Required, Default="acme.json"_ | ||||
|  | ||||
| The `storage` option sets the location where your ACME certificates are saved to. | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| [certificatesResolvers.myresolver.acme] | ||||
|   # ... | ||||
|   storage = "acme.json" | ||||
|   # ... | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| certificatesResolvers: | ||||
|   myresolver: | ||||
| @@ -463,6 +562,13 @@ certificatesResolvers: | ||||
|       # ... | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| [certificatesResolvers.myresolver.acme] | ||||
|   # ... | ||||
|   storage = "acme.json" | ||||
|   # ... | ||||
| ``` | ||||
|  | ||||
| ```bash tab="CLI" | ||||
| # ... | ||||
| --certificatesresolvers.myresolver.acme.storage=acme.json | ||||
| @@ -484,13 +590,118 @@ docker run -v "/my/host/acme:/etc/traefik/acme" traefik | ||||
| !!! warning | ||||
|     For concurrency reasons, this file cannot be shared across multiple instances of Traefik. | ||||
|  | ||||
| ### `certificatesDuration` | ||||
|  | ||||
| _Optional, Default=2160_ | ||||
|  | ||||
| The `certificatesDuration` option defines the certificates' duration in hours. | ||||
| It defaults to `2160` (90 days) to follow Let's Encrypt certificates' duration. | ||||
|  | ||||
| !!! warning "Traefik cannot manage certificates with a duration lower than 1 hour." | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| certificatesResolvers: | ||||
|   myresolver: | ||||
|     acme: | ||||
|       # ... | ||||
|       certificatesDuration: 72 | ||||
|       # ... | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| [certificatesResolvers.myresolver.acme] | ||||
|   # ... | ||||
|   certificatesDuration=72 | ||||
|   # ... | ||||
| ``` | ||||
|  | ||||
| ```bash tab="CLI" | ||||
| # ... | ||||
| --certificatesresolvers.myresolver.acme.certificatesduration=72 | ||||
| # ... | ||||
| ``` | ||||
|  | ||||
| `certificatesDuration` is used to calculate two durations: | ||||
|  | ||||
| - `Renew Period`: the period before the end of the certificate duration, during which the certificate should be renewed. | ||||
| - `Renew Interval`: the interval between renew attempts. | ||||
|  | ||||
| | Certificate Duration | Renew Period      | Renew Interval          | | ||||
| |----------------------|-------------------|-------------------------| | ||||
| | >= 1 year            | 4 months          | 1 week                  | | ||||
| | >= 90 days           | 30 days           | 1 day                   | | ||||
| | >= 7 days            | 1 day             | 1 hour                  | | ||||
| | >= 24 hours          | 6 hours           | 10 min                  | | ||||
| | < 24 hours           | 20 min            | 1 min                   | | ||||
|  | ||||
| ### `preferredChain` | ||||
|  | ||||
| _Optional, Default=""_ | ||||
|  | ||||
| Preferred chain to use. | ||||
|  | ||||
| If the CA offers multiple certificate chains, prefer the chain with an issuer matching this Subject Common Name. | ||||
| If no match, the default offered chain will be used. | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| certificatesResolvers: | ||||
|   myresolver: | ||||
|     acme: | ||||
|       # ... | ||||
|       preferredChain: 'ISRG Root X1' | ||||
|       # ... | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| [certificatesResolvers.myresolver.acme] | ||||
|   # ... | ||||
|   preferredChain = "ISRG Root X1" | ||||
|   # ... | ||||
| ``` | ||||
|  | ||||
| ```bash tab="CLI" | ||||
| # ... | ||||
| --certificatesresolvers.myresolver.acme.preferredChain=ISRG Root X1 | ||||
| # ... | ||||
| ``` | ||||
|  | ||||
| ### `keyType` | ||||
|  | ||||
| _Optional, Default="RSA4096"_ | ||||
|  | ||||
| KeyType used for generating certificate private key. Allow value 'EC256', 'EC384', 'RSA2048', 'RSA4096', 'RSA8192'. | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| certificatesResolvers: | ||||
|   myresolver: | ||||
|     acme: | ||||
|       # ... | ||||
|       keyType: 'RSA4096' | ||||
|       # ... | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| [certificatesResolvers.myresolver.acme] | ||||
|   # ... | ||||
|   keyType = "RSA4096" | ||||
|   # ... | ||||
| ``` | ||||
|  | ||||
| ```bash tab="CLI" | ||||
| # ... | ||||
| --certificatesresolvers.myresolver.acme.keyType=RSA4096 | ||||
| # ... | ||||
| ``` | ||||
|  | ||||
| ## Fallback | ||||
|  | ||||
| If Let's Encrypt is not reachable, the following certificates will apply: | ||||
|  | ||||
|   1. Previously generated ACME certificates (before downtime) | ||||
|   1. Expired ACME certificates | ||||
|   1. Provided certificates | ||||
|   2. Expired ACME certificates | ||||
|   3. Provided certificates | ||||
|  | ||||
| !!! important | ||||
|     For new (sub)domains which need Let's Encrypt authentication, the default Traefik certificate will be used until Traefik is restarted. | ||||
|  | ||||
| {!traefik-for-business-applications.md!} | ||||
|   | ||||
| @@ -22,7 +22,7 @@ deploy: | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: IngressRoute | ||||
| metadata: | ||||
|   name: blogtls | ||||
| @@ -64,18 +64,6 @@ labels: | ||||
|   - traefik.http.routers.blog.tls.domains[0].sans=*.example.org | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| ## Dynamic configuration | ||||
| [http.routers] | ||||
|   [http.routers.blog] | ||||
|     rule = "Host(`example.com`) && Path(`/blog`)" | ||||
|     [http.routers.blog.tls] | ||||
|       certResolver = "myresolver" # From static configuration | ||||
|       [[http.routers.blog.tls.domains]] | ||||
|         main = "example.org" | ||||
|         sans = ["*.example.org"] | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| ## Dynamic configuration | ||||
| http: | ||||
| @@ -89,3 +77,15 @@ http: | ||||
|             sans: | ||||
|               - "*.example.org" | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| ## Dynamic configuration | ||||
| [http.routers] | ||||
|   [http.routers.blog] | ||||
|     rule = "Host(`example.com`) && Path(`/blog`)" | ||||
|     [http.routers.blog.tls] | ||||
|       certResolver = "myresolver" # From static configuration | ||||
|       [[http.routers.blog.tls.domains]] | ||||
|         main = "example.org" | ||||
|         sans = ["*.example.org"] | ||||
| ``` | ||||
|   | ||||
| @@ -18,7 +18,7 @@ deploy: | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: IngressRoute | ||||
| metadata: | ||||
|   name: blogtls | ||||
| @@ -52,15 +52,6 @@ labels: | ||||
|   - traefik.http.routers.blog.tls.certresolver=myresolver | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| ## Dynamic configuration | ||||
| [http.routers] | ||||
|   [http.routers.blog] | ||||
|     rule = "(Host(`example.com`) && Path(`/blog`)) || Host(`blog.example.org`)" | ||||
|     [http.routers.blog.tls] | ||||
|       certResolver = "myresolver" | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| ## Dynamic configuration | ||||
| http: | ||||
| @@ -70,3 +61,12 @@ http: | ||||
|       tls: | ||||
|         certResolver: myresolver | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| ## Dynamic configuration | ||||
| [http.routers] | ||||
|   [http.routers.blog] | ||||
|     rule = "(Host(`example.com`) && Path(`/blog`)) || Host(`blog.example.org`)" | ||||
|     [http.routers.blog.tls] | ||||
|       certResolver = "myresolver" | ||||
| ``` | ||||
|   | ||||
| @@ -18,7 +18,7 @@ deploy: | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: IngressRoute | ||||
| metadata: | ||||
|   name: blogtls | ||||
| @@ -52,15 +52,6 @@ labels: | ||||
|   - traefik.http.routers.blog.tls.certresolver=myresolver | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| ## Dynamic configuration | ||||
| [http.routers] | ||||
|   [http.routers.blog] | ||||
|   rule = "Host(`example.com`) && Path(`/blog`)" | ||||
|   [http.routers.blog.tls] | ||||
|     certResolver = "myresolver" | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| ## Dynamic configuration | ||||
| http: | ||||
| @@ -70,3 +61,12 @@ http: | ||||
|       tls: | ||||
|         certResolver: myresolver | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| ## Dynamic configuration | ||||
| [http.routers] | ||||
|   [http.routers.blog] | ||||
|   rule = "Host(`example.com`) && Path(`/blog`)" | ||||
|   [http.routers.blog.tls] | ||||
|     certResolver = "myresolver" | ||||
| ``` | ||||
|   | ||||
| @@ -1,3 +1,8 @@ | ||||
| --- | ||||
| title: "Traefik Proxy  HTTPS & TLS Overview |Traefik Docs" | ||||
| description: "Traefik supports HTTPS & TLS, which concerns roughly two parts of the configuration: routers, and the TLS connection. Read the documentation to learn more." | ||||
| --- | ||||
|  | ||||
| # HTTPS & TLS | ||||
|  | ||||
| Overview | ||||
| @@ -14,3 +19,5 @@ The next sections of this documentation explain how to configure the TLS connect | ||||
| That is to say, how to obtain [TLS certificates](./tls.md#certificates-definition): | ||||
| either through a definition in the dynamic configuration, or through [Let's Encrypt](./acme.md) (ACME). | ||||
| And how to configure [TLS options](./tls.md#tls-options), and [certificates stores](./tls.md#certificates-stores). | ||||
|  | ||||
| {!traefik-for-business-applications.md!} | ||||
|   | ||||
| @@ -22,6 +22,24 @@ | ||||
|   # | ||||
|   # caServer = "https://acme-staging-v02.api.letsencrypt.org/directory" | ||||
|  | ||||
|   # The certificates' duration in hours. | ||||
|   # It defaults to 2160 (90 days) to follow Let's Encrypt certificates' duration. | ||||
|   # | ||||
|   # Optional | ||||
|   # Default: 2160 | ||||
|   # | ||||
|   # certificatesDuration=2160 | ||||
|  | ||||
|   # Preferred chain to use. | ||||
|   # | ||||
|   # If the CA offers multiple certificate chains, prefer the chain with an issuer matching this Subject Common Name. | ||||
|   # If no match, the default offered chain will be used. | ||||
|   # | ||||
|   # Optional | ||||
|   # Default: "" | ||||
|   # | ||||
|   # preferredChain = "ISRG Root X1" | ||||
|  | ||||
|   # KeyType to use. | ||||
|   # | ||||
|   # Optional | ||||
|   | ||||
| @@ -21,6 +21,24 @@ | ||||
| # | ||||
| --certificatesresolvers.myresolver.acme.caserver=https://acme-staging-v02.api.letsencrypt.org/directory | ||||
|  | ||||
| # The certificates' duration in hours. | ||||
| # It defaults to 2160 (90 days) to follow Let's Encrypt certificates' duration. | ||||
| # | ||||
| # Optional | ||||
| # Default: 2160 | ||||
| # | ||||
| --certificatesresolvers.myresolver.acme.certificatesDuration=2160 | ||||
|  | ||||
| # Preferred chain to use. | ||||
| # | ||||
| # If the CA offers multiple certificate chains, prefer the chain with an issuer matching this Subject Common Name. | ||||
| # If no match, the default offered chain will be used. | ||||
| # | ||||
| # Optional | ||||
| # Default: "" | ||||
| # | ||||
| --certificatesresolvers.myresolver.acme.preferredchain="ISRG Root X1" | ||||
|  | ||||
| # KeyType to use. | ||||
| # | ||||
| # Optional | ||||
|   | ||||
| @@ -24,6 +24,24 @@ certificatesResolvers: | ||||
|       # | ||||
|       # caServer: "https://acme-staging-v02.api.letsencrypt.org/directory" | ||||
|  | ||||
|       # The certificates' duration in hours. | ||||
|       # It defaults to 2160 (90 days) to follow Let's Encrypt certificates' duration. | ||||
|       # | ||||
|       # Optional | ||||
|       # Default: 2160 | ||||
|       # | ||||
|       # certificatesDuration: 2160 | ||||
|  | ||||
|       # Preferred chain to use. | ||||
|       # | ||||
|       # If the CA offers multiple certificate chains, prefer the chain with an issuer matching this Subject Common Name. | ||||
|       # If no match, the default offered chain will be used. | ||||
|       # | ||||
|       # Optional | ||||
|       # Default: "" | ||||
|       # | ||||
|       # preferredChain: 'ISRG Root X1' | ||||
|  | ||||
|       # KeyType to use. | ||||
|       # | ||||
|       # Optional | ||||
|   | ||||
| @@ -1,3 +1,8 @@ | ||||
| --- | ||||
| title: "Traefik TLS Documentation" | ||||
| description: "Learn how to configure the transport layer security (TLS) connection in Traefik Proxy. Read the technical documentation." | ||||
| --- | ||||
|  | ||||
| # TLS | ||||
|  | ||||
| Transport Layer Security | ||||
| @@ -13,18 +18,6 @@ See the [Let's Encrypt](./acme.md) page. | ||||
|  | ||||
| To add / remove TLS certificates, even when Traefik is already running, their definition can be added to the [dynamic configuration](../getting-started/configuration-overview.md), in the `[[tls.certificates]]` section: | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [[tls.certificates]] | ||||
|   certFile = "/path/to/domain.cert" | ||||
|   keyFile = "/path/to/domain.key" | ||||
|  | ||||
| [[tls.certificates]] | ||||
|   certFile = "/path/to/other-domain.cert" | ||||
|   keyFile = "/path/to/other-domain.key" | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| @@ -36,23 +29,28 @@ tls: | ||||
|       keyFile: /path/to/other-domain.key | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [[tls.certificates]] | ||||
|   certFile = "/path/to/domain.cert" | ||||
|   keyFile = "/path/to/domain.key" | ||||
|  | ||||
| [[tls.certificates]] | ||||
|   certFile = "/path/to/other-domain.cert" | ||||
|   keyFile = "/path/to/other-domain.key" | ||||
| ``` | ||||
|  | ||||
| !!! important "Restriction" | ||||
|  | ||||
|     In the above example, we've used the [file provider](../providers/file.md) to handle these definitions. | ||||
|     It is the only available method to configure the certificates (as well as the options and the stores). | ||||
|     However, in [Kubernetes](../providers/kubernetes-crd.md), the certificates can and must be provided by [secrets](https://kubernetes.io/docs/concepts/configuration/secret/).  | ||||
|     However, in [Kubernetes](../providers/kubernetes-crd.md), the certificates can and must be provided by [secrets](https://kubernetes.io/docs/concepts/configuration/secret/). | ||||
|  | ||||
| ## Certificates Stores | ||||
|  | ||||
| In Traefik, certificates are grouped together in certificates stores, which are defined as such: | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.stores] | ||||
|   [tls.stores.default] | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| @@ -61,28 +59,20 @@ tls: | ||||
|     default: {} | ||||
| ``` | ||||
|  | ||||
| !!! important "Restriction" | ||||
|  | ||||
|     Any store definition other than the default one (named `default`) will be ignored, | ||||
|     and there is thefore only one globally available TLS store. | ||||
|  | ||||
| In the `tls.certificates` section, a list of stores can then be specified to indicate where the certificates should be stored: | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [[tls.certificates]] | ||||
|   certFile = "/path/to/domain.cert" | ||||
|   keyFile = "/path/to/domain.key" | ||||
|   stores = ["default"] | ||||
|  | ||||
| [[tls.certificates]] | ||||
|   # Note that since no store is defined, | ||||
|   # the certificate below will be stored in the `default` store. | ||||
|   certFile = "/path/to/other-domain.cert" | ||||
|   keyFile = "/path/to/other-domain.key" | ||||
| [tls.stores] | ||||
|   [tls.stores.default] | ||||
| ``` | ||||
|  | ||||
| !!! important "Restriction" | ||||
|  | ||||
|     Any store definition other than the default one (named `default`) will be ignored, | ||||
|     and there is therefore only one globally available TLS store. | ||||
|  | ||||
| In the `tls.certificates` section, a list of stores can then be specified to indicate where the certificates should be stored: | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| @@ -98,6 +88,21 @@ tls: | ||||
|       keyFile: /path/to/other-domain.key | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [[tls.certificates]] | ||||
|   certFile = "/path/to/domain.cert" | ||||
|   keyFile = "/path/to/domain.key" | ||||
|   stores = ["default"] | ||||
|  | ||||
| [[tls.certificates]] | ||||
|   # Note that since no store is defined, | ||||
|   # the certificate below will be stored in the `default` store. | ||||
|   certFile = "/path/to/other-domain.cert" | ||||
|   keyFile = "/path/to/other-domain.key" | ||||
| ``` | ||||
|  | ||||
| !!! important "Restriction" | ||||
|  | ||||
|     The `stores` list will actually be ignored and automatically set to `["default"]`. | ||||
| @@ -107,16 +112,6 @@ tls: | ||||
| Traefik can use a default certificate for connections without a SNI, or without a matching domain. | ||||
| This default certificate should be defined in a TLS store: | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.stores] | ||||
|   [tls.stores.default] | ||||
|     [tls.stores.default.defaultCertificate] | ||||
|       certFile = "path/to/cert.crt" | ||||
|       keyFile  = "path/to/cert.key" | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| @@ -128,40 +123,135 @@ tls: | ||||
|         keyFile: path/to/cert.key | ||||
| ``` | ||||
|  | ||||
| If no default certificate is provided, Traefik generates and uses a self-signed certificate. | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| ## Domain fronting | ||||
| [tls.stores] | ||||
|   [tls.stores.default] | ||||
|     [tls.stores.default.defaultCertificate] | ||||
|       certFile = "path/to/cert.crt" | ||||
|       keyFile  = "path/to/cert.key" | ||||
| ``` | ||||
|  | ||||
| Basically, [domain fronting](https://en.wikipedia.org/wiki/Domain_fronting) is a technique that allows to open a  | ||||
| connection with a specific domain name, thanks to the  | ||||
| [Server Name Indication](https://en.wikipedia.org/wiki/Server_Name_Indication), then access a service with another  | ||||
| domain set in the HTTP `Host` header. | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSStore | ||||
| metadata: | ||||
|   name: default | ||||
|   namespace: default | ||||
|  | ||||
| Since the `v2.2.2`, Traefik avoids (by default) using domain fronting. | ||||
| As it is valid for advanced use cases, the `HostHeader` and `HostSNI` [rules](../routing/routers/index.md#rule) allow  | ||||
| to fine tune the routing with the `Server Name Indication` and `Host header` value. | ||||
| spec: | ||||
|   defaultCertificate: | ||||
|     secretName: default-certificate | ||||
|      | ||||
| --- | ||||
| apiVersion: v1 | ||||
| kind: Secret | ||||
| metadata: | ||||
|   name: default-certificate | ||||
|   namespace: default | ||||
|    | ||||
| type: Opaque | ||||
| data: | ||||
|   tls.crt: LS0tLS1CRUdJTiBDRVJUSUZJQ0FURS0tLS0tCi0tLS0tRU5EIENFUlRJRklDQVRFLS0tLS0= | ||||
|   tls.key: LS0tLS1CRUdJTiBQUklWQVRFIEtFWS0tLS0tCi0tLS0tRU5EIFBSSVZBVEUgS0VZLS0tLS0= | ||||
| ``` | ||||
|  | ||||
| If you encounter routing issues with a previously working configuration, please refer to the  | ||||
| [migration guide](../migration/v2.md) to update your configuration. | ||||
| If no `defaultCertificate` is provided, Traefik will use the generated one. | ||||
|  | ||||
| ### ACME Default Certificate | ||||
|  | ||||
| You can configure Traefik to use an ACME provider (like Let's Encrypt) to generate the default certificate. | ||||
| The configuration to resolve the default certificate should be defined in a TLS store: | ||||
|  | ||||
| !!! important "Precedence with the `defaultGeneratedCert` option" | ||||
|  | ||||
|     The `defaultGeneratedCert` definition takes precedence over the ACME default certificate configuration. | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| tls: | ||||
|   stores: | ||||
|     default: | ||||
|       defaultGeneratedCert: | ||||
|         resolver: myresolver | ||||
|         domain: | ||||
|           main: example.org | ||||
|           sans: | ||||
|             - foo.example.org | ||||
|             - bar.example.org | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.stores] | ||||
|   [tls.stores.default.defaultGeneratedCert] | ||||
|     resolver = "myresolver" | ||||
|     [tls.stores.default.defaultGeneratedCert.domain] | ||||
|       main = "example.org" | ||||
|       sans = ["foo.example.org", "bar.example.org"] | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSStore | ||||
| metadata: | ||||
|   name: default | ||||
|   namespace: default | ||||
|  | ||||
| spec: | ||||
|   defaultGeneratedCert: | ||||
|     resolver: myresolver | ||||
|     domain: | ||||
|       main: example.org | ||||
|       sans: | ||||
|         - foo.example.org | ||||
|         - bar.example.org | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Docker" | ||||
| ## Dynamic configuration | ||||
| labels: | ||||
|   - "traefik.tls.stores.default.defaultgeneratedcert.resolver=myresolver" | ||||
|   - "traefik.tls.stores.default.defaultgeneratedcert.domain.main=example.org" | ||||
|   - "traefik.tls.stores.default.defaultgeneratedcert.domain.sans=foo.example.org, bar.example.org" | ||||
| ``` | ||||
|  | ||||
| ```json tab="Marathon" | ||||
| labels: { | ||||
|   "traefik.tls.stores.default.defaultgeneratedcert.resolver": "myresolver", | ||||
|   "traefik.tls.stores.default.defaultgeneratedcert.domain.main": "example.org", | ||||
|   "traefik.tls.stores.default.defaultgeneratedcert.domain.sans": "foo.example.org, bar.example.org", | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## TLS Options | ||||
|  | ||||
| The TLS options allow one to configure some parameters of the TLS connection. | ||||
|  | ||||
| !!! important "'default' TLS Option" | ||||
|  | ||||
|     The `default` option is special. | ||||
|     When no tls options are specified in a tls router, the `default` option is used.   | ||||
|     When specifying the `default` option explicitly, make sure not to specify provider namespace as the `default` option does not have one.   | ||||
|     Conversely, for cross-provider references, for example, when referencing the file provider from a docker label, | ||||
|     you must specify the provider namespace, for example:   | ||||
|     `traefik.http.routers.myrouter.tls.options=myoptions@file` | ||||
|  | ||||
| !!! important "TLSOption in Kubernetes" | ||||
|  | ||||
|     When using the [TLSOption resource](../../routing/providers/kubernetes-crd#kind-tlsoption) in Kubernetes, one might setup a default set of options that, | ||||
|     if not explicitly overwritten, should apply to all ingresses.   | ||||
|     To achieve that, you'll have to create a TLSOption resource with the name `default`. | ||||
|     There may exist only one TLSOption with the name `default` (across all namespaces) - otherwise they will be dropped.   | ||||
|     To explicitly use a different TLSOption (and using the Kubernetes Ingress resources) | ||||
|     you'll have to add an annotation to the Ingress in the following form: | ||||
|     `traefik.ingress.kubernetes.io/router.tls.options: <resource-namespace>-<resource-name>@kubernetescrd` | ||||
|  | ||||
| ### Minimum TLS Version | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|  | ||||
|   [tls.options.default] | ||||
|     minVersion = "VersionTLS12" | ||||
|  | ||||
|   [tls.options.mintls13] | ||||
|     minVersion = "VersionTLS13" | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| @@ -174,8 +264,20 @@ tls: | ||||
|       minVersion: VersionTLS13 | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|  | ||||
|   [tls.options.default] | ||||
|     minVersion = "VersionTLS12" | ||||
|  | ||||
|   [tls.options.mintls13] | ||||
|     minVersion = "VersionTLS13" | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: default | ||||
| @@ -185,7 +287,7 @@ spec: | ||||
|   minVersion: VersionTLS12 | ||||
|  | ||||
| --- | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: mintls13 | ||||
| @@ -197,21 +299,9 @@ spec: | ||||
|  | ||||
| ### Maximum TLS Version | ||||
|  | ||||
| We discourages the use of this setting to disable TLS1.3. | ||||
| We discourage the use of this setting to disable TLS1.3. | ||||
|  | ||||
| The right approach is to update the clients to support TLS1.3. | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|  | ||||
|   [tls.options.default] | ||||
|     maxVersion = "VersionTLS13" | ||||
|  | ||||
|   [tls.options.maxtls12] | ||||
|     maxVersion = "VersionTLS12" | ||||
| ``` | ||||
| The recommended approach is to update the clients to support TLS1.3. | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
| @@ -225,8 +315,20 @@ tls: | ||||
|       maxVersion: VersionTLS12 | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|  | ||||
|   [tls.options.default] | ||||
|     maxVersion = "VersionTLS13" | ||||
|  | ||||
|   [tls.options.maxtls12] | ||||
|     maxVersion = "VersionTLS12" | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: default | ||||
| @@ -236,7 +338,7 @@ spec: | ||||
|   maxVersion: VersionTLS13 | ||||
|  | ||||
| --- | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: maxtls12 | ||||
| @@ -250,16 +352,6 @@ spec: | ||||
|  | ||||
| See [cipherSuites](https://godoc.org/crypto/tls#pkg-constants) for more information. | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|   [tls.options.default] | ||||
|     cipherSuites = [ | ||||
|       "TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256" | ||||
|     ] | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| @@ -270,8 +362,18 @@ tls: | ||||
|         - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|   [tls.options.default] | ||||
|     cipherSuites = [ | ||||
|       "TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256" | ||||
|     ] | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: default | ||||
| @@ -296,14 +398,6 @@ The names of the curves defined by [`crypto`](https://godoc.org/crypto/tls#Curve | ||||
|  | ||||
| See [CurveID](https://godoc.org/crypto/tls#CurveID) for more information. | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|   [tls.options.default] | ||||
|     curvePreferences = ["CurveP521", "CurveP384"] | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| @@ -315,8 +409,16 @@ tls: | ||||
|         - CurveP384 | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|   [tls.options.default] | ||||
|     curvePreferences = ["CurveP521", "CurveP384"] | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: default | ||||
| @@ -330,16 +432,9 @@ spec: | ||||
|  | ||||
| ### Strict SNI Checking | ||||
|  | ||||
| With strict SNI checking, Traefik won't allow connections from clients connections | ||||
| that do not specify a server_name extension or don't match any certificate configured on the tlsOption. | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|   [tls.options.default] | ||||
|     sniStrict = true | ||||
| ``` | ||||
| With strict SNI checking enabled, Traefik won't allow connections from clients that do not specify a server_name extension | ||||
| or don't match any of the configured certificates. | ||||
| The default certificate is irrelevant on that matter. | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
| @@ -350,8 +445,16 @@ tls: | ||||
|       sniStrict: true | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|   [tls.options.default] | ||||
|     sniStrict = true | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: default | ||||
| @@ -361,18 +464,14 @@ spec: | ||||
|   sniStrict: true | ||||
| ``` | ||||
|  | ||||
| ### Prefer Server Cipher Suites | ||||
| ### ALPN Protocols | ||||
|  | ||||
| This option allows the server to choose its most preferred cipher suite instead of the client's. | ||||
| Please note that this is enabled automatically when `minVersion` or `maxVersion` are set. | ||||
| _Optional, Default="h2, http/1.1, acme-tls/1"_ | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|   [tls.options.default] | ||||
|     preferServerCipherSuites = true | ||||
| ``` | ||||
| This option allows to specify the list of supported application level protocols for the TLS handshake, | ||||
| in order of preference. | ||||
| If the client supports ALPN, the selected protocol will be one from this list,  | ||||
| and the connection will fail if there is no mutually supported protocol. | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
| @@ -380,45 +479,48 @@ Please note that this is enabled automatically when `minVersion` or `maxVersion` | ||||
| tls: | ||||
|   options: | ||||
|     default: | ||||
|       preferServerCipherSuites: true | ||||
|       alpnProtocols: | ||||
|         - http/1.1 | ||||
|         - h2 | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: default | ||||
|   namespace: default | ||||
|  | ||||
| spec: | ||||
|   preferServerCipherSuites: true | ||||
| ``` | ||||
|  | ||||
| ### Client Authentication (mTLS) | ||||
|  | ||||
| Traefik supports mutual authentication, through the `clientAuth` section. | ||||
|  | ||||
| For authentication policies that require verification of the client certificate, the certificate authority for the certificate should be set in `clientAuth.caFiles`. | ||||
|   | ||||
| The `clientAuth.clientAuthType` option governs the behaviour as follows: | ||||
|  | ||||
| - `NoClientCert`: disregards any client certificate. | ||||
| - `RequestClientCert`: asks for a certificate but proceeds anyway if none is provided. | ||||
| - `RequireAnyClientCert`: requires a certificate but does not verify if it is signed by a CA listed in `clientAuth.caFiles`. | ||||
| - `VerifyClientCertIfGiven`: if a certificate is provided, verifies if it is signed by a CA listed in `clientAuth.caFiles`. Otherwise proceeds without any certificate. | ||||
| - `RequireAndVerifyClientCert`: requires a certificate, which must be signed by a CA listed in `clientAuth.caFiles`.  | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|   [tls.options.default] | ||||
|     [tls.options.default.clientAuth] | ||||
|       # in PEM format. each file can contain multiple CAs. | ||||
|       caFiles = ["tests/clientca1.crt", "tests/clientca2.crt"] | ||||
|       clientAuthType = "RequireAndVerifyClientCert" | ||||
|     alpnProtocols = ["http/1.1", "h2"] | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: default | ||||
|   namespace: default | ||||
|  | ||||
| spec: | ||||
|   alpnProtocols: | ||||
|     - http/1.1 | ||||
|     - h2 | ||||
| ``` | ||||
|  | ||||
| ### Client Authentication (mTLS) | ||||
|  | ||||
| Traefik supports mutual authentication, through the `clientAuth` section. | ||||
|  | ||||
| For authentication policies that require verification of the client certificate, the certificate authority for the certificates should be set in `clientAuth.caFiles`. | ||||
|  | ||||
| In Kubernetes environment, CA certificate can be set in `clientAuth.secretNames`. See [TLSOption resource](../../routing/providers/kubernetes-crd#kind-tlsoption) for more details. | ||||
|  | ||||
| The `clientAuth.clientAuthType` option governs the behaviour as follows: | ||||
|  | ||||
| - `NoClientCert`: disregards any client certificate. | ||||
| - `RequestClientCert`: asks for a certificate but proceeds anyway if none is provided. | ||||
| - `RequireAnyClientCert`: requires a certificate but does not verify if it is signed by a CA listed in `clientAuth.caFiles` or in `clientAuth.secretNames`. | ||||
| - `VerifyClientCertIfGiven`: if a certificate is provided, verifies if it is signed by a CA listed in `clientAuth.caFiles` or in `clientAuth.secretNames`. Otherwise proceeds without any certificate. | ||||
| - `RequireAndVerifyClientCert`: requires a certificate, which must be signed by a CA listed in `clientAuth.caFiles` or in `clientAuth.secretNames`. | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| @@ -433,8 +535,19 @@ tls: | ||||
|         clientAuthType: RequireAndVerifyClientCert | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Dynamic configuration | ||||
|  | ||||
| [tls.options] | ||||
|   [tls.options.default] | ||||
|     [tls.options.default.clientAuth] | ||||
|       # in PEM format. each file can contain multiple CAs. | ||||
|       caFiles = ["tests/clientca1.crt", "tests/clientca2.crt"] | ||||
|       clientAuthType = "RequireAndVerifyClientCert" | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: TLSOption | ||||
| metadata: | ||||
|   name: default | ||||
| @@ -442,7 +555,10 @@ metadata: | ||||
|  | ||||
| spec: | ||||
|   clientAuth: | ||||
|     # the CA certificate is extracted from key `tls.ca` or `ca.crt` of the given secrets. | ||||
|     secretNames: | ||||
|       - secretCA | ||||
|     clientAuthType: RequireAndVerifyClientCert | ||||
| ``` | ||||
|  | ||||
| {!traefik-for-business-applications.md!} | ||||
|   | ||||
							
								
								
									
										4
									
								
								docs/content/includes/.markdownlint.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								docs/content/includes/.markdownlint.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| { | ||||
|   "extends": "../../.markdownlint.json", | ||||
|   "MD041": false | ||||
| } | ||||
							
								
								
									
										14
									
								
								docs/content/includes/traefik-for-business-applications.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								docs/content/includes/traefik-for-business-applications.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| --- | ||||
|  | ||||
| !!! question "Using Traefik for Business Applications?" | ||||
|  | ||||
|     If you are using Traefik in your organization, consider our enterprise-grade solutions: | ||||
|  | ||||
|     - API Management   | ||||
|       [Explore](https://traefik.io/solutions/api-management/) // [Watch Demo Video](https://info.traefik.io/watch-traefik-hub-demo) | ||||
|     - API Gateway   | ||||
|       [Explore](https://traefik.io/solutions/api-gateway/) // [Watch Demo Video](https://info.traefik.io/watch-traefikee-demo) | ||||
|     - Ingress Controller   | ||||
|       [Kubernetes](https://traefik.io/solutions/kubernetes-ingress/) // [Docker Swarm](https://traefik.io/solutions/docker-swarm-ingress/) | ||||
|  | ||||
|     These tools help businesses discover, deploy, secure, and manage microservices and APIs easily, at scale, across any environment. | ||||
| @@ -1,9 +1,13 @@ | ||||
| --- | ||||
| title: "Traefik Proxy Documentation" | ||||
| description: "Traefik Proxy, an open source Edge Router, auto-discovers configurations and supports major orchestrators, like Kubernetes. Read the technical documentation." | ||||
| --- | ||||
|  | ||||
| # Welcome | ||||
|  | ||||
|  | ||||
|  | ||||
| Traefik is an [open-source](https://github.com/containous/traefik) *Edge Router* that makes publishing your services a fun and easy experience.  | ||||
| Traefik is an [open-source](https://github.com/traefik/traefik) *Edge Router* that makes publishing your services a fun and easy experience.  | ||||
| It receives requests on behalf of your system and finds out which components are responsible for handling them.  | ||||
|  | ||||
| What sets Traefik apart, besides its many features, is that it automatically discovers the right configuration for your services.  | ||||
| @@ -14,15 +18,14 @@ Traefik is natively compliant with every major cluster technology, such as Kuber | ||||
| With Traefik, there is no need to maintain and synchronize a separate configuration file: everything happens automatically, in real time (no restarts, no connection interruptions). | ||||
| With Traefik, you spend time developing and deploying new features to your system, not on configuring and maintaining its working state.    | ||||
|  | ||||
| Developing Traefik, our main goal is to make it simple to use, and we're sure you'll enjoy it. | ||||
| Developing Traefik, our main goal is to make it effortless to use, and we're sure you'll enjoy it. | ||||
|  | ||||
| -- The Traefik Maintainer Team  | ||||
|  | ||||
| !!! info | ||||
|  | ||||
|     Join our user friendly and active [Community Forum](https://community.containo.us) to discuss, learn, and connect with the traefik community. | ||||
|      | ||||
|     If you're a business running critical services behind Traefik, | ||||
|     know that [Containous](https://containo.us), the company that sponsors Traefik's development, | ||||
|     can provide [commercial support](https://info.containo.us/commercial-services) | ||||
|     and develops an [Enterprise Edition](https://containo.us/traefikee/) of Traefik. | ||||
|     Join our user friendly and active [Community Forum](https://community.traefik.io "Link to Traefik Community Forum") to discuss, learn, and connect with the traefik community. | ||||
|  | ||||
|     Using Traefik in your organization? Consider [Traefik Enterprise](https://traefik.io/traefik-enterprise/ "Lino to Traefik Enterprise"), our unified API Gateway and Ingress that simplifies the discovery, security, and deployment of APIs and microservices across any environment. | ||||
|  | ||||
|     See it in action in [this short video walkthrough](https://info.traefik.io/watch-traefikee-demo "Link to video walkthrough"). | ||||
|   | ||||
| @@ -1,112 +0,0 @@ | ||||
| # ErrorPage | ||||
|  | ||||
| It Has Never Been Easier to Say That Something Went Wrong | ||||
| {: .subtitle } | ||||
|  | ||||
|  | ||||
|  | ||||
| The ErrorPage middleware returns a custom page in lieu of the default, according to configured ranges of HTTP Status codes. | ||||
|  | ||||
| !!! important | ||||
|     The error page itself is _not_ hosted by Traefik. | ||||
|  | ||||
| ## Configuration Examples | ||||
|  | ||||
| ```yaml tab="Docker" | ||||
| # Dynamic Custom Error Page for 5XX Status Code | ||||
| labels: | ||||
|   - "traefik.http.middlewares.test-errorpage.errors.status=500-599" | ||||
|   - "traefik.http.middlewares.test-errorpage.errors.service=serviceError" | ||||
|   - "traefik.http.middlewares.test-errorpage.errors.query=/{status}.html" | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: test-errorpage | ||||
| spec: | ||||
|   errors: | ||||
|     status: | ||||
|       - 500-599 | ||||
|     query: /{status}.html | ||||
|     service: | ||||
|       name: whoami | ||||
|       port: 80 | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Consul Catalog" | ||||
| # Dynamic Custom Error Page for 5XX Status Code | ||||
| - "traefik.http.middlewares.test-errorpage.errors.status=500-599" | ||||
| - "traefik.http.middlewares.test-errorpage.errors.service=serviceError" | ||||
| - "traefik.http.middlewares.test-errorpage.errors.query=/{status}.html" | ||||
| ``` | ||||
|  | ||||
| ```json tab="Marathon" | ||||
| "labels": { | ||||
|   "traefik.http.middlewares.test-errorpage.errors.status": "500-599", | ||||
|   "traefik.http.middlewares.test-errorpage.errors.service": "serviceError", | ||||
|   "traefik.http.middlewares.test-errorpage.errors.query": "/{status}.html" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="Rancher" | ||||
| # Dynamic Custom Error Page for 5XX Status Code | ||||
| labels: | ||||
|   - "traefik.http.middlewares.test-errorpage.errors.status=500-599" | ||||
|   - "traefik.http.middlewares.test-errorpage.errors.service=serviceError" | ||||
|   - "traefik.http.middlewares.test-errorpage.errors.query=/{status}.html" | ||||
| ``` | ||||
|  | ||||
| ```toml tab="File (TOML)" | ||||
| # Custom Error Page for 5XX | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-errorpage.errors] | ||||
|     status = ["500-599"] | ||||
|     service = "serviceError" | ||||
|     query = "/{status}.html" | ||||
|  | ||||
| [http.services] | ||||
|   # ... definition of error-handler-service and my-service | ||||
| ``` | ||||
|  | ||||
| ```yaml tab="File (YAML)" | ||||
| # Custom Error Page for 5XX | ||||
| http: | ||||
|   middlewares: | ||||
|     test-errorpage: | ||||
|       errors: | ||||
|         status: | ||||
|           - "500-599" | ||||
|         service: serviceError | ||||
|         query: "/{status}.html" | ||||
|  | ||||
| [http.services] | ||||
|   # ... definition of error-handler-service and my-service | ||||
| ``` | ||||
|  | ||||
| !!! note ""  | ||||
|     In this example, the error page URL is based on the status code (`query=/{status}.html`). | ||||
|  | ||||
| ## Configuration Options | ||||
|  | ||||
| ### `status` | ||||
|  | ||||
| The `status` that will trigger the error page. | ||||
|  | ||||
| The status code ranges are inclusive (`500-599` will trigger with every code between `500` and `599`, `500` and `599` included). | ||||
|   | ||||
| !!! note ""  | ||||
|  | ||||
|     You can define either a status code like `500` or ranges with a syntax like `500-599`. | ||||
|  | ||||
| ### `service` | ||||
|  | ||||
| The service that will serve the new requested error page. | ||||
|  | ||||
| !!! note ""  | ||||
|     In kubernetes, you need to reference a kubernetes service instead of a traefik service. | ||||
|  | ||||
| ### `query` | ||||
|  | ||||
| The URL for the error page (hosted by `service`). You can use `{status}` in the query, that will be replaced by the received status code. | ||||
| @@ -1,11 +1,16 @@ | ||||
| --- | ||||
| title: "Traefik AddPrefix Documentation" | ||||
| description: "Learn how to implement the HTTP AddPrefix middleware in Traefik Proxy to updates request paths before being forwarded. Read the technical documentation." | ||||
| --- | ||||
| 
 | ||||
| # Add Prefix | ||||
| 
 | ||||
| Prefixing the Path  | ||||
| Prefixing the Path | ||||
| {: .subtitle } | ||||
| 
 | ||||
|   | ||||
|  | ||||
| 
 | ||||
| The AddPrefix middleware updates the URL Path of the request before forwarding it. | ||||
| The AddPrefix middleware updates the path of a request before forwarding it. | ||||
| 
 | ||||
| ## Configuration Examples | ||||
| 
 | ||||
| @@ -17,7 +22,7 @@ labels: | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| # Prefixing with /foo | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: add-foo | ||||
| @@ -43,13 +48,6 @@ labels: | ||||
|   - "traefik.http.middlewares.add-foo.addprefix.prefix=/foo" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Prefixing with /foo | ||||
| [http.middlewares] | ||||
|   [http.middlewares.add-foo.addPrefix] | ||||
|     prefix = "/foo" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| # Prefixing with /foo | ||||
| http: | ||||
| @@ -59,9 +57,16 @@ http: | ||||
|         prefix: "/foo" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Prefixing with /foo | ||||
| [http.middlewares] | ||||
|   [http.middlewares.add-foo.addPrefix] | ||||
|     prefix = "/foo" | ||||
| ``` | ||||
| 
 | ||||
| ## Configuration Options | ||||
| 
 | ||||
| ### `prefix` | ||||
| 
 | ||||
| `prefix` is the string to add before the current path in the requested URL. | ||||
| It should include the leading slash (`/`). | ||||
| It should include a leading slash (`/`). | ||||
| @@ -1,11 +1,16 @@ | ||||
| --- | ||||
| title: "Traefik BasicAuth Documentation" | ||||
| description: "The HTTP basic authentication (BasicAuth) middleware in Traefik Proxy restricts access to your Services to known users. Read the technical documentation." | ||||
| --- | ||||
| 
 | ||||
| # BasicAuth | ||||
| 
 | ||||
| Adding Basic Authentication | ||||
| {: .subtitle } | ||||
| 
 | ||||
|  | ||||
|  | ||||
| 
 | ||||
| The BasicAuth middleware is a quick way to restrict access to your services to known users. | ||||
| The BasicAuth middleware grants access to services to authorized users only. | ||||
| 
 | ||||
| ## Configuration Examples | ||||
| 
 | ||||
| @@ -14,7 +19,7 @@ The BasicAuth middleware is a quick way to restrict access to your services to k | ||||
| # | ||||
| # Note: when used in docker-compose.yml all dollar signs in the hash need to be doubled for escaping. | ||||
| # To create user:password pair, it's possible to use this command: | ||||
| # echo $(htpasswd -nb user password) | sed -e s/\\$/\\$\\$/g | ||||
| # echo $(htpasswd -nB user) | sed -e s/\\$/\\$\\$/g | ||||
| # | ||||
| # Also note that dollar signs should NOT be doubled when they not evaluated (e.g. Ansible docker_container module). | ||||
| labels: | ||||
| @@ -23,7 +28,7 @@ labels: | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| # Declaring the user list | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: test-auth | ||||
| @@ -48,16 +53,6 @@ labels: | ||||
|   - "traefik.http.middlewares.test-auth.basicauth.users=test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/,test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Declaring the user list | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|   users = [ | ||||
|     "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/",  | ||||
|     "test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0", | ||||
|   ] | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| # Declaring the user list | ||||
| http: | ||||
| @@ -65,42 +60,61 @@ http: | ||||
|     test-auth: | ||||
|       basicAuth: | ||||
|         users: | ||||
|           - "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/"  | ||||
|           - "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/" | ||||
|           - "test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Declaring the user list | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|   users = [ | ||||
|     "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/", | ||||
|     "test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0", | ||||
|   ] | ||||
| ``` | ||||
| 
 | ||||
| ## Configuration Options | ||||
| 
 | ||||
| ### General | ||||
| 
 | ||||
| Passwords must be hashed using MD5, SHA1, or BCrypt. | ||||
| 
 | ||||
| !!! tip  | ||||
| !!! tip | ||||
| 
 | ||||
|     Use `htpasswd` to generate the passwords. | ||||
| 
 | ||||
| ### `users` | ||||
| 
 | ||||
| The `users` option is an array of authorized users. Each user will be declared using the `name:hashed-password` format. | ||||
| The `users` option is an array of authorized users. Each user must be declared using the `name:hashed-password` format. | ||||
| 
 | ||||
| !!! note "" | ||||
|      | ||||
| 
 | ||||
|     - If both `users` and `usersFile` are provided, the two are merged. The contents of `usersFile` have precedence over the values in `users`. | ||||
|     - For security reasons, the field `users` doesn't exist for Kubernetes IngressRoute, and one should use the `secret` field instead. | ||||
| 
 | ||||
| !!! note "Kubernetes kubernetes.io/basic-auth secret type" | ||||
|      | ||||
|     Kubernetes supports a special `kubernetes.io/basic-auth` secret type. | ||||
|     This secret must contain two keys: `username` and `password`. | ||||
|     Please note that these keys are not hashed or encrypted in any way, and therefore is less secure than other methods. | ||||
|     You can find more information on the [Kubernetes Basic Authentication Secret Documentation](https://kubernetes.io/docs/concepts/configuration/secret/#basic-authentication-secret) | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| # Declaring the user list | ||||
| # | ||||
| # Note: all dollar signs in the hash need to be doubled for escaping. | ||||
| # Note: when used in docker-compose.yml all dollar signs in the hash need to be doubled for escaping. | ||||
| # To create a user:password pair, the following command can be used: | ||||
| # echo $(htpasswd -nb user password) | sed -e s/\\$/\\$\\$/g | ||||
| # | ||||
| # Also note that dollar signs should NOT be doubled when they not evaluated (e.g. Ansible docker_container module). | ||||
| labels: | ||||
|   - "traefik.http.middlewares.test-auth.basicauth.users=test:$$apr1$$H6uskkkW$$IgXLP6ewTrSuBkTrqE8wj/,test2:$$apr1$$d9hr9HBB$$4HxwgUir3HP4EsggP/QNo0" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| # Declaring the user list | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: test-auth | ||||
| @@ -118,11 +132,24 @@ kind: Secret | ||||
| metadata: | ||||
|   name: authsecret | ||||
|   namespace: default | ||||
| 
 | ||||
| data: | ||||
|   users: |2 | ||||
|     dGVzdDokYXByMSRINnVza2trVyRJZ1hMUDZld1RyU3VCa1RycUU4d2ovCnRlc3QyOiRhcHIxJGQ5 | ||||
|     aHI5SEJCJDRIeHdnVWlyM0hQNEVzZ2dQL1FObzAK | ||||
| 
 | ||||
| --- | ||||
| # This is an alternate auth secret that demonstrates the basic-auth secret type. | ||||
| # Note: the password is not hashed, and is merely base64 encoded. | ||||
| 
 | ||||
| apiVersion: v1 | ||||
| kind: Secret | ||||
| metadata: | ||||
|   name: authsecret2 | ||||
|   namespace: default | ||||
| type: kubernetes.io/basic-auth | ||||
| data: | ||||
|   username: dXNlcg== # username: user | ||||
|   password: cGFzc3dvcmQ= # password: password | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Consul Catalog" | ||||
| @@ -142,16 +169,6 @@ labels: | ||||
|   - "traefik.http.middlewares.test-auth.basicauth.users=test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/,test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Declaring the user list | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|     users = [ | ||||
|       "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/",  | ||||
|       "test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0", | ||||
|     ] | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| # Declaring the user list | ||||
| http: | ||||
| @@ -159,10 +176,20 @@ http: | ||||
|     test-auth: | ||||
|       basicAuth: | ||||
|         users: | ||||
|           - "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/"  | ||||
|           - "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/" | ||||
|           - "test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Declaring the user list | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|     users = [ | ||||
|       "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/", | ||||
|       "test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0", | ||||
|     ] | ||||
| ``` | ||||
| 
 | ||||
| ### `usersFile` | ||||
| 
 | ||||
| The `usersFile` option is the path to an external file that contains the authorized users for the middleware. | ||||
| @@ -170,9 +197,9 @@ The `usersFile` option is the path to an external file that contains the authori | ||||
| The file content is a list of `name:hashed-password`. | ||||
| 
 | ||||
| !!! note "" | ||||
|      | ||||
| 
 | ||||
|     - If both `users` and `usersFile` are provided, the two are merged. The contents of `usersFile` have precedence over the values in `users`. | ||||
|     - Because it does not make much sense to refer to a file path on Kubernetes, the `usersFile` field doesn't exist for Kubernetes IngressRoute, and one should use the `secret` field instead.  | ||||
|     - Because it does not make much sense to refer to a file path on Kubernetes, the `usersFile` field doesn't exist for Kubernetes IngressRoute, and one should use the `secret` field instead. | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| labels: | ||||
| @@ -180,7 +207,7 @@ labels: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: test-auth | ||||
| @@ -216,12 +243,6 @@ labels: | ||||
|   - "traefik.http.middlewares.test-auth.basicauth.usersfile=/path/to/my/usersfile" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|     usersFile = "/path/to/my/usersfile" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
| @@ -230,6 +251,12 @@ http: | ||||
|         usersFile: "/path/to/my/usersfile" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|     usersFile = "/path/to/my/usersfile" | ||||
| ``` | ||||
| 
 | ||||
| ??? example "A file containing test/test and test2/test2" | ||||
| 
 | ||||
|     ```txt | ||||
| @@ -239,7 +266,7 @@ http: | ||||
| 
 | ||||
| ### `realm` | ||||
| 
 | ||||
| You can customize the realm for the authentication with the `realm` option. The default value is `traefik`.  | ||||
| You can customize the realm for the authentication with the `realm` option. The default value is `traefik`. | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| labels: | ||||
| @@ -247,7 +274,7 @@ labels: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: test-auth | ||||
| @@ -271,12 +298,6 @@ labels: | ||||
|   - "traefik.http.middlewares.test-auth.basicauth.realm=MyRealm" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|     realm = "MyRealm" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
| @@ -285,6 +306,12 @@ http: | ||||
|         realm: "MyRealm" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|     realm = "MyRealm" | ||||
| ``` | ||||
| 
 | ||||
| ### `headerField` | ||||
| 
 | ||||
| You can define a header field to store the authenticated user using the `headerField`option. | ||||
| @@ -295,7 +322,7 @@ labels: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: my-auth | ||||
| @@ -315,12 +342,6 @@ spec: | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares.my-auth.basicAuth] | ||||
|   # ... | ||||
|   headerField = "X-WebAuth-User" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
| @@ -330,6 +351,12 @@ http: | ||||
|         headerField: "X-WebAuth-User" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares.my-auth.basicAuth] | ||||
|   # ... | ||||
|   headerField = "X-WebAuth-User" | ||||
| ``` | ||||
| 
 | ||||
| ### `removeHeader` | ||||
| 
 | ||||
| Set the `removeHeader` option to `true` to remove the authorization header before forwarding the request to your service. (Default value is `false`.) | ||||
| @@ -340,7 +367,7 @@ labels: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: test-auth | ||||
| @@ -364,12 +391,6 @@ labels: | ||||
|   - "traefik.http.middlewares.test-auth.basicauth.removeheader=true" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|     removeHeader = true | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
| @@ -377,3 +398,9 @@ http: | ||||
|       basicAuth: | ||||
|         removeHeader: true | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-auth.basicAuth] | ||||
|     removeHeader = true | ||||
| ``` | ||||
| @@ -1,27 +1,32 @@ | ||||
| --- | ||||
| title: "Traefik Buffering Documentation" | ||||
| description: "The HTTP buffering middleware in Traefik Proxy limits the size of requests that can be forwarded to Services. Read the technical documentation." | ||||
| --- | ||||
| 
 | ||||
| # Buffering | ||||
| 
 | ||||
| How to Read the Request before Forwarding It | ||||
| {: .subtitle } | ||||
| 
 | ||||
|  | ||||
|  | ||||
| 
 | ||||
| The Buffering middleware gives you control on how you want to read the requests before sending them to services. | ||||
| The Buffering middleware limits the size of requests that can be forwarded to services. | ||||
| 
 | ||||
| With Buffering, Traefik reads the entire request into memory (possibly buffering large requests into disk), and rejects requests that are over a specified limit. | ||||
| With Buffering, Traefik reads the entire request into memory (possibly buffering large requests into disk), and rejects requests that are over a specified size limit. | ||||
| 
 | ||||
| This can help services deal with large data (multipart/form-data for example), and can minimize time spent sending data to a service. | ||||
| This can help services avoid large amounts of data (`multipart/form-data` for example), and can minimize the time spent sending data to a service. | ||||
| 
 | ||||
| ## Configuration Examples | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| # Sets the maximum request body to 2Mb | ||||
| # Sets the maximum request body to 2MB | ||||
| labels: | ||||
|   - "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| # Sets the maximum request body to 2Mb | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| # Sets the maximum request body to 2MB | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: limit | ||||
| @@ -31,7 +36,7 @@ spec: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Consul Catalog" | ||||
| # Sets the maximum request body to 2Mb | ||||
| # Sets the maximum request body to 2MB | ||||
| - "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000" | ||||
| ``` | ||||
| 
 | ||||
| @@ -42,20 +47,13 @@ spec: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Rancher" | ||||
| # Sets the maximum request body to 2Mb | ||||
| # Sets the maximum request body to 2MB | ||||
| labels: | ||||
|   - "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Sets the maximum request body to 2Mb | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     maxRequestBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| # Sets the maximum request body to 2Mb | ||||
| # Sets the maximum request body to 2MB | ||||
| http: | ||||
|   middlewares: | ||||
|     limit: | ||||
| @@ -63,13 +61,22 @@ http: | ||||
|         maxRequestBodyBytes: 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Sets the maximum request body to 2MB | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     maxRequestBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ## Configuration Options | ||||
| 
 | ||||
| ### `maxRequestBodyBytes` | ||||
| 
 | ||||
| With the `maxRequestBodyBytes` option, you can configure the maximum allowed body size for the request (in Bytes). | ||||
| _Optional, Default=0_ | ||||
| 
 | ||||
| If the request exceeds the allowed size, it is not forwarded to the service and the client gets a `413 (Request Entity Too Large)` response. | ||||
| The `maxRequestBodyBytes` option configures the maximum allowed body size for the request (in bytes). | ||||
| 
 | ||||
| If the request exceeds the allowed size, it is not forwarded to the service, and the client gets a `413` (Request Entity Too Large) response. | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| labels: | ||||
| @@ -77,7 +84,7 @@ labels: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: limit | ||||
| @@ -101,12 +108,6 @@ labels: | ||||
|   - "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     maxRequestBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
| @@ -115,9 +116,17 @@ http: | ||||
|         maxRequestBodyBytes: 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     maxRequestBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ### `memRequestBodyBytes` | ||||
| 
 | ||||
| You can configure a threshold (in Bytes) from which the request will be buffered on disk instead of in memory with the `memRequestBodyBytes` option.  | ||||
| _Optional, Default=1048576_ | ||||
| 
 | ||||
| You can configure a threshold (in bytes) from which the request will be buffered on disk instead of in memory with the `memRequestBodyBytes` option. | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| labels: | ||||
| @@ -125,7 +134,7 @@ labels: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: limit | ||||
| @@ -149,12 +158,6 @@ labels: | ||||
|   - "traefik.http.middlewares.limit.buffering.memRequestBodyBytes=2000000" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     memRequestBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
| @@ -163,11 +166,19 @@ http: | ||||
|         memRequestBodyBytes: 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     memRequestBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ### `maxResponseBodyBytes` | ||||
| 
 | ||||
| With the `maxReesponseBodyBytes` option, you can configure the maximum allowed response size from the service (in Bytes). | ||||
| _Optional, Default=0_ | ||||
| 
 | ||||
| If the response exceeds the allowed size, it is not forwarded to the client. The client gets a `413 (Request Entity Too Large) response` instead. | ||||
| The `maxResponseBodyBytes` option configures the maximum allowed response size from the service (in bytes). | ||||
| 
 | ||||
| If the response exceeds the allowed size, it is not forwarded to the client. The client gets a `500` (Internal Server Error) response instead. | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| labels: | ||||
| @@ -175,7 +186,7 @@ labels: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: limit | ||||
| @@ -199,12 +210,6 @@ labels: | ||||
|   - "traefik.http.middlewares.limit.buffering.maxResponseBodyBytes=2000000" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     maxResponseBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
| @@ -213,9 +218,17 @@ http: | ||||
|         maxResponseBodyBytes: 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     maxResponseBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ### `memResponseBodyBytes` | ||||
| 
 | ||||
| You can configure a threshold (in Bytes) from which the response will be buffered on disk instead of in memory with the `memResponseBodyBytes` option.  | ||||
| _Optional, Default=1048576_ | ||||
| 
 | ||||
| You can configure a threshold (in bytes) from which the response will be buffered on disk instead of in memory with the `memResponseBodyBytes` option. | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| labels: | ||||
| @@ -223,7 +236,7 @@ labels: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: limit | ||||
| @@ -247,12 +260,6 @@ labels: | ||||
|   - "traefik.http.middlewares.limit.buffering.memResponseBodyBytes=2000000" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     memResponseBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
| @@ -261,19 +268,27 @@ http: | ||||
|         memResponseBodyBytes: 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.limit.buffering] | ||||
|     memResponseBodyBytes = 2000000 | ||||
| ``` | ||||
| 
 | ||||
| ### `retryExpression` | ||||
| 
 | ||||
| You can have the Buffering middleware replay the request with the help of the `retryExpression` option. | ||||
| _Optional, Default=""_ | ||||
| 
 | ||||
| You can have the Buffering middleware replay the request using `retryExpression`. | ||||
| 
 | ||||
| ??? example "Retries once in the case of a network error" | ||||
| 
 | ||||
| ??? example "Retries once in case of a network error" | ||||
|      | ||||
|     ```yaml tab="Docker" | ||||
|     labels: | ||||
|       - "traefik.http.middlewares.limit.buffering.retryExpression=IsNetworkError() && Attempts() < 2" | ||||
|     ``` | ||||
|      | ||||
| 
 | ||||
|     ```yaml tab="Kubernetes" | ||||
|     apiVersion: traefik.containo.us/v1alpha1 | ||||
|     apiVersion: traefik.io/v1alpha1 | ||||
|     kind: Middleware | ||||
|     metadata: | ||||
|       name: limit | ||||
| @@ -281,28 +296,22 @@ You can have the Buffering middleware replay the request with the help of the `r | ||||
|       buffering: | ||||
|         retryExpression: "IsNetworkError() && Attempts() < 2" | ||||
|     ``` | ||||
|      | ||||
| 
 | ||||
|     ```yaml tab="Consul Catalog" | ||||
|     - "traefik.http.middlewares.limit.buffering.retryExpression=IsNetworkError() && Attempts() < 2" | ||||
|     ``` | ||||
|          | ||||
| 
 | ||||
|     ```json tab="Marathon" | ||||
|     "labels": { | ||||
|       "traefik.http.middlewares.limit.buffering.retryExpression": "IsNetworkError() && Attempts() < 2" | ||||
|     } | ||||
|     ``` | ||||
|      | ||||
| 
 | ||||
|     ```yaml tab="Rancher" | ||||
|     labels: | ||||
|       - "traefik.http.middlewares.limit.buffering.retryExpression=IsNetworkError() && Attempts() < 2" | ||||
|     ``` | ||||
|      | ||||
|     ```toml tab="File (TOML)" | ||||
|     [http.middlewares] | ||||
|       [http.middlewares.limit.buffering] | ||||
|         retryExpression = "IsNetworkError() && Attempts() < 2" | ||||
|     ``` | ||||
|      | ||||
| 
 | ||||
|     ```yaml tab="File (YAML)" | ||||
|     http: | ||||
|       middlewares: | ||||
| @@ -311,8 +320,14 @@ You can have the Buffering middleware replay the request with the help of the `r | ||||
|             retryExpression: "IsNetworkError() && Attempts() < 2" | ||||
|     ``` | ||||
| 
 | ||||
|     ```toml tab="File (TOML)" | ||||
|     [http.middlewares] | ||||
|       [http.middlewares.limit.buffering] | ||||
|         retryExpression = "IsNetworkError() && Attempts() < 2" | ||||
|     ``` | ||||
| 
 | ||||
| The retry expression is defined as a logical combination of the functions below with the operators AND (`&&`) and OR (`||`). At least one function is required: | ||||
| 
 | ||||
| - `Attempts()` number of attempts (the first one counts) | ||||
| - `ResponseCode()` response code of the service | ||||
| - `IsNetworkError()` - if the response code is related to networking error  | ||||
| - `IsNetworkError()` whether the response code is related to networking error | ||||
| @@ -1,16 +1,21 @@ | ||||
| --- | ||||
| title: "Traefik Command Line Documentation" | ||||
| description: "The HTTP chain middleware lets you define reusable combinations of other middleware, to reuse the same groups. Read the technical documentation." | ||||
| --- | ||||
| 
 | ||||
| # Chain | ||||
| 
 | ||||
| When One Isn't Enough | ||||
| {: .subtitle } | ||||
| 
 | ||||
|  | ||||
|  | ||||
| 
 | ||||
| The Chain middleware enables you to define reusable combinations of other pieces of middleware.  | ||||
| The Chain middleware enables you to define reusable combinations of other pieces of middleware. | ||||
| It makes reusing the same groups easier. | ||||
| 
 | ||||
| ## Configuration Example | ||||
| 
 | ||||
| Example "A Chain for WhiteList, BasicAuth, and HTTPS" | ||||
| Below is an example of a Chain containing `WhiteList`, `BasicAuth`, and `RedirectScheme`. | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| labels: | ||||
| @@ -21,20 +26,18 @@ labels: | ||||
|   - "traefik.http.middlewares.auth-users.basicauth.users=test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/" | ||||
|   - "traefik.http.middlewares.https-only.redirectscheme.scheme=https" | ||||
|   - "traefik.http.middlewares.known-ips.ipwhitelist.sourceRange=192.168.1.7,127.0.0.1/32" | ||||
|   - "http.services.service1.loadbalancer.server.port=80" | ||||
|   - "traefik.http.services.service1.loadbalancer.server.port=80" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: IngressRoute | ||||
| metadata: | ||||
|   name: test | ||||
|   namespace: default | ||||
| 
 | ||||
| spec: | ||||
|   entryPoints: | ||||
|     - web | ||||
| 
 | ||||
|   routes: | ||||
|     - match: Host(`mydomain`) | ||||
|       kind: Rule | ||||
| @@ -44,7 +47,7 @@ spec: | ||||
|       middlewares: | ||||
|         - name: secured | ||||
| --- | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: secured | ||||
| @@ -55,7 +58,7 @@ spec: | ||||
|     - name: known-ips | ||||
|     - name: auth-users | ||||
| --- | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: auth-users | ||||
| @@ -64,7 +67,7 @@ spec: | ||||
|     users: | ||||
|     - test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/ | ||||
| --- | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: https-only | ||||
| @@ -72,7 +75,7 @@ spec: | ||||
|   redirectScheme: | ||||
|     scheme: https | ||||
| --- | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: known-ips | ||||
| @@ -91,7 +94,7 @@ spec: | ||||
| - "traefik.http.middlewares.auth-users.basicauth.users=test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/" | ||||
| - "traefik.http.middlewares.https-only.redirectscheme.scheme=https" | ||||
| - "traefik.http.middlewares.known-ips.ipwhitelist.sourceRange=192.168.1.7,127.0.0.1/32" | ||||
| - "http.services.service1.loadbalancer.server.port=80" | ||||
| - "traefik.http.services.service1.loadbalancer.server.port=80" | ||||
| ``` | ||||
| 
 | ||||
| ```json tab="Marathon" | ||||
| @@ -103,7 +106,7 @@ spec: | ||||
|   "traefik.http.middlewares.auth-users.basicauth.users": "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/", | ||||
|   "traefik.http.middlewares.https-only.redirectscheme.scheme": "https", | ||||
|   "traefik.http.middlewares.known-ips.ipwhitelist.sourceRange": "192.168.1.7,127.0.0.1/32", | ||||
|   "http.services.service1.loadbalancer.server.port": "80" | ||||
|   "traefik.http.services.service1.loadbalancer.server.port": "80" | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| @@ -116,39 +119,11 @@ labels: | ||||
|   - "traefik.http.middlewares.auth-users.basicauth.users=test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/" | ||||
|   - "traefik.http.middlewares.https-only.redirectscheme.scheme=https" | ||||
|   - "traefik.http.middlewares.known-ips.ipwhitelist.sourceRange=192.168.1.7,127.0.0.1/32" | ||||
|   - "http.services.service1.loadbalancer.server.port=80" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # ...     | ||||
| [http.routers] | ||||
|   [http.routers.router1] | ||||
|     service = "service1" | ||||
|     middlewares = ["secured"] | ||||
|     rule = "Host(`mydomain`)" | ||||
| 
 | ||||
| [http.middlewares] | ||||
|   [http.middlewares.secured.chain] | ||||
|     middlewares = ["https-only", "known-ips", "auth-users"] | ||||
| 
 | ||||
|   [http.middlewares.auth-users.basicAuth] | ||||
|     users = ["test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/"] | ||||
| 
 | ||||
|   [http.middlewares.https-only.redirectScheme] | ||||
|     scheme = "https" | ||||
| 
 | ||||
|   [http.middlewares.known-ips.ipWhiteList] | ||||
|     sourceRange = ["192.168.1.7", "127.0.0.1/32"] | ||||
| 
 | ||||
| [http.services] | ||||
|   [http.services.service1] | ||||
|     [http.services.service1.loadBalancer] | ||||
|       [[http.services.service1.loadBalancer.servers]] | ||||
|         url = "http://127.0.0.1:80" | ||||
|   - "traefik.http.services.service1.loadbalancer.server.port=80" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| # ...     | ||||
| # ... | ||||
| http: | ||||
|   routers: | ||||
|     router1: | ||||
| @@ -186,3 +161,31 @@ http: | ||||
|         servers: | ||||
|           - url: "http://127.0.0.1:80" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # ... | ||||
| [http.routers] | ||||
|   [http.routers.router1] | ||||
|     service = "service1" | ||||
|     middlewares = ["secured"] | ||||
|     rule = "Host(`mydomain`)" | ||||
| 
 | ||||
| [http.middlewares] | ||||
|   [http.middlewares.secured.chain] | ||||
|     middlewares = ["https-only", "known-ips", "auth-users"] | ||||
| 
 | ||||
|   [http.middlewares.auth-users.basicAuth] | ||||
|     users = ["test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/"] | ||||
| 
 | ||||
|   [http.middlewares.https-only.redirectScheme] | ||||
|     scheme = "https" | ||||
| 
 | ||||
|   [http.middlewares.known-ips.ipWhiteList] | ||||
|     sourceRange = ["192.168.1.7", "127.0.0.1/32"] | ||||
| 
 | ||||
| [http.services] | ||||
|   [http.services.service1] | ||||
|     [http.services.service1.loadBalancer] | ||||
|       [[http.services.service1.loadBalancer.servers]] | ||||
|         url = "http://127.0.0.1:80" | ||||
| ``` | ||||
| @@ -1,29 +1,31 @@ | ||||
| --- | ||||
| title: "Traefik CircuitBreaker Documentation" | ||||
| description: "The HTTP circuit breaker in Traefik Proxy prevents stacking requests to unhealthy Services, resulting in cascading failures. Read the technical documentation." | ||||
| --- | ||||
| 
 | ||||
| # CircuitBreaker | ||||
| 
 | ||||
| Don't Waste Time Calling Unhealthy Services | ||||
| {: .subtitle } | ||||
| 
 | ||||
|   | ||||
|  | ||||
| 
 | ||||
| The circuit breaker protects your system from stacking requests to unhealthy services (resulting in cascading failures). | ||||
| The circuit breaker protects your system from stacking requests to unhealthy services, resulting in cascading failures. | ||||
| 
 | ||||
| When your system is healthy, the circuit is close (normal operations).  | ||||
| When your system becomes unhealthy, the circuit becomes open and the requests are no longer forwarded (but handled by a fallback mechanism). | ||||
| When your system is healthy, the circuit is closed (normal operations). | ||||
| When your system becomes unhealthy, the circuit opens, and the requests are no longer forwarded, but instead are handled by a fallback mechanism. | ||||
| 
 | ||||
| To assess if your system is healthy, the circuit breaker constantly monitors the services.  | ||||
| To assess if your system is healthy, the circuit breaker constantly monitors the services. | ||||
| 
 | ||||
| !!! note "" | ||||
| 
 | ||||
|     - The CircuitBreaker only analyses what happens _after_ it is positioned in the middleware chain. What happens _before_ has no impact on its state. | ||||
|     - The CircuitBreaker only affects the routers that use it. Routers that don't use the CircuitBreaker won't be affected by its state. | ||||
|     The CircuitBreaker only analyzes what happens _after_ its position within the middleware chain. What happens _before_ has no impact on its state. | ||||
| 
 | ||||
| !!! important | ||||
| 
 | ||||
|     Each router will eventually gets its own instance of a given circuit breaker. | ||||
|      | ||||
|     If two different routers refer to the same circuit breaker definition, they will get one instance each. | ||||
|     It means that one circuit breaker can be open while the other stays closed: their state is not shared. | ||||
|      | ||||
|     Each router gets its own instance of a given circuit breaker. | ||||
|     One circuit breaker instance can be open while the other remains closed: their state is not shared. | ||||
| 
 | ||||
|     This is the expected behavior, we want you to be able to define what makes a service healthy without having to declare a circuit breaker for each route. | ||||
| 
 | ||||
| ## Configuration Examples | ||||
| @@ -36,7 +38,7 @@ labels: | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| # Latency Check | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: latency-check | ||||
| @@ -62,13 +64,6 @@ labels: | ||||
|   - "traefik.http.middlewares.latency-check.circuitbreaker.expression=LatencyAtQuantileMS(50.0) > 100" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Latency Check | ||||
| [http.middlewares] | ||||
|   [http.middlewares.latency-check.circuitBreaker] | ||||
|     expression = "LatencyAtQuantileMS(50.0) > 100" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| # Latency Check | ||||
| http: | ||||
| @@ -78,82 +73,90 @@ http: | ||||
|         expression: "LatencyAtQuantileMS(50.0) > 100" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Latency Check | ||||
| [http.middlewares] | ||||
|   [http.middlewares.latency-check.circuitBreaker] | ||||
|     expression = "LatencyAtQuantileMS(50.0) > 100" | ||||
| ``` | ||||
| 
 | ||||
| ## Possible States | ||||
| 
 | ||||
| There are three possible states for your circuit breaker: | ||||
| 
 | ||||
| - Close (your service operates normally) | ||||
| - Closed (your service operates normally) | ||||
| - Open (the fallback mechanism takes over your service) | ||||
| - Recovering (the circuit breaker tries to resume normal operations by progressively sending requests to your service) | ||||
| 
 | ||||
| ### Close | ||||
| ### Closed | ||||
| 
 | ||||
| While close, the circuit breaker only collects metrics to analyze the behavior of the requests. | ||||
| While the circuit is closed, the circuit breaker only collects metrics to analyze the behavior of the requests. | ||||
| 
 | ||||
| At specified intervals (`checkPeriod`), it will evaluate `expression` to decide if its state must change.  | ||||
| At specified intervals (`checkPeriod`), the circuit breaker evaluates `expression` to decide if its state must change. | ||||
| 
 | ||||
| ### Open | ||||
| 
 | ||||
| While open, the fallback mechanism takes over the normal service calls for a duration of `FallbackDuration`. | ||||
| After this duration, it will enter the recovering state. | ||||
| After this duration, it enters the recovering state. | ||||
| 
 | ||||
| ### Recovering | ||||
| 
 | ||||
| While recovering, the circuit breaker will progressively send requests to your service again (in a linear way, for `RecoveryDuration`). | ||||
| If your service fails during recovery, the circuit breaker becomes open again. | ||||
| If the service operates normally during the whole recovering duration, then the circuit breaker returns to close. | ||||
| While recovering, the circuit breaker sends linearly increasing amounts of requests to your service (for `RecoveryDuration`). | ||||
| If your service fails during recovery, the circuit breaker opens again. | ||||
| If the service operates normally during the entire recovery duration, then the circuit breaker closes. | ||||
| 
 | ||||
| ## Configuration Options | ||||
| 
 | ||||
| ### Configuring the Trigger | ||||
| 
 | ||||
| You can specify an `expression` that, once matched, will trigger the circuit breaker (and apply the fallback mechanism instead of calling your services). | ||||
| You can specify an `expression` that, once matched, opens the circuit breaker and applies the fallback mechanism instead of calling your services. | ||||
| 
 | ||||
| The `expression` can check three different metrics: | ||||
| The `expression` option can check three different metrics: | ||||
| 
 | ||||
| - The network error ratio (`NetworkErrorRatio`) | ||||
| - The status code ratio (`ResponseCodeRatio`) | ||||
| - The latency at quantile, in milliseconds (`LatencyAtQuantileMS`) | ||||
| - The latency at a quantile in milliseconds (`LatencyAtQuantileMS`) | ||||
| 
 | ||||
| #### `NetworkErrorRatio` | ||||
| 
 | ||||
| If you want the circuit breaker to trigger at a 30% ratio of network errors, the expression will be `NetworkErrorRatio() > 0.30` | ||||
| If you want the circuit breaker to open at a 30% ratio of network errors, the `expression` is `NetworkErrorRatio() > 0.30` | ||||
| 
 | ||||
| #### `ResponseCodeRatio` | ||||
| 
 | ||||
| You can trigger the circuit breaker based on the ratio of a given range of status codes. | ||||
| You can configure the circuit breaker to open based on the ratio of a given range of status codes. | ||||
| 
 | ||||
| The `ResponseCodeRatio` accepts four parameters, `from`, `to`, `dividedByFrom`, `dividedByTo`. | ||||
| 
 | ||||
| The operation that will be computed is sum(`to` -> `from`) / sum (`dividedByFrom` -> `dividedByTo`). | ||||
| 
 | ||||
| !!! note "" | ||||
|     If sum (`dividedByFrom` -> `dividedByTo`) equals 0, then `ResponseCodeRatio` returns 0. | ||||
|      | ||||
|     `from`is inclusive, `to` is exclusive.  | ||||
| 
 | ||||
| For example, the expression `ResponseCodeRatio(500, 600, 0, 600) > 0.25` will trigger the circuit breaker if 25% of the requests returned a 5XX status (amongst the request that returned a status code from 0 to 5XX).  | ||||
|     If sum (`dividedByFrom` -> `dividedByTo`) equals 0, then `ResponseCodeRatio` returns 0. | ||||
| 
 | ||||
|     `from`is inclusive, `to` is exclusive. | ||||
| 
 | ||||
| For example, the expression `ResponseCodeRatio(500, 600, 0, 600) > 0.25` will trigger the circuit breaker if 25% of the requests returned a 5XX status (amongst the request that returned a status code from 0 to 5XX). | ||||
| 
 | ||||
| #### `LatencyAtQuantileMS` | ||||
| 
 | ||||
| You can trigger the circuit breaker when a given proportion of your requests become too slow. | ||||
| You can configure the circuit breaker to open when a given proportion of your requests become too slow. | ||||
| 
 | ||||
| For example, the expression `LatencyAtQuantileMS(50.0) > 100` will trigger the circuit breaker when the median latency (quantile 50) reaches 100MS. | ||||
| For example, the expression `LatencyAtQuantileMS(50.0) > 100` opens the circuit breaker when the median latency (quantile 50) reaches 100ms. | ||||
| 
 | ||||
| !!! note "" | ||||
| 
 | ||||
|     You must provide a float number (with the trailing .0) for the quantile value | ||||
|   | ||||
| #### Using multiple metrics | ||||
|     You must provide a floating point number (with the trailing .0) for the quantile value | ||||
| 
 | ||||
| You can combine multiple metrics using operators in your expression. | ||||
| #### Using Multiple Metrics | ||||
| 
 | ||||
| You can combine multiple metrics using operators in your `expression`. | ||||
| 
 | ||||
| Supported operators are: | ||||
| 
 | ||||
| - AND (`&&`) | ||||
| - OR (`||`) | ||||
| 
 | ||||
| For example, `ResponseCodeRatio(500, 600, 0, 600) > 0.30 || NetworkErrorRatio() > 0.10` triggers the circuit breaker when 30% of the requests return a 5XX status code, or when the ratio of network errors reaches 10%.  | ||||
| For example, `ResponseCodeRatio(500, 600, 0, 600) > 0.30 || NetworkErrorRatio() > 0.10` triggers the circuit breaker when 30% of the requests return a 5XX status code, or when the ratio of network errors reaches 10%. | ||||
| 
 | ||||
| #### Operators | ||||
| 
 | ||||
| @@ -168,20 +171,23 @@ Here is the list of supported operators: | ||||
| 
 | ||||
| ### Fallback mechanism | ||||
| 
 | ||||
| The fallback mechanism returns a `HTTP 503 Service Unavailable` to the client (instead of calling the target service). | ||||
| This behavior cannot be configured.  | ||||
| The fallback mechanism returns a `HTTP 503 Service Unavailable` to the client instead of calling the target service. | ||||
| This behavior cannot be configured. | ||||
| 
 | ||||
| ### `CheckPeriod` | ||||
| 
 | ||||
| The interval used to evaluate `expression` and decide if the state of the circuit breaker must change. | ||||
| By default, `CheckPeriod` is 100ms. This value cannot be configured. | ||||
| _Optional, Default="100ms"_ | ||||
| 
 | ||||
| The interval between successive checks of the circuit breaker condition (when in standby state). | ||||
| 
 | ||||
| ### `FallbackDuration` | ||||
| 
 | ||||
| By default, `FallbackDuration` is 10 seconds. This value cannot be configured. | ||||
| _Optional, Default="10s"_ | ||||
| 
 | ||||
| ### `RecoveringDuration` | ||||
| The duration for which the circuit breaker will wait before trying to recover (from a tripped state). | ||||
| 
 | ||||
| The duration of the recovering mode (recovering state).  | ||||
| ### `RecoveryDuration` | ||||
| 
 | ||||
| By default, `RecoveringDuration` is 10 seconds. This value cannot be configured.   | ||||
| _Optional, Default="10s"_ | ||||
| 
 | ||||
| The duration for which the circuit breaker will try to recover (as soon as it is in recovering state). | ||||
| @@ -1,11 +1,16 @@ | ||||
| --- | ||||
| title: "Traefik Compress Documentation" | ||||
| description: "Traefik Proxy's HTTP middleware lets you compress responses before sending them to the client. Read the technical documentation." | ||||
| --- | ||||
| 
 | ||||
| # Compress | ||||
| 
 | ||||
| Compressing the Response before Sending it to the Client | ||||
| Compress Responses before Sending them to the Client | ||||
| {: .subtitle } | ||||
| 
 | ||||
|  | ||||
|  | ||||
| 
 | ||||
| The Compress middleware enables the gzip compression.  | ||||
| The Compress middleware uses gzip compression. | ||||
| 
 | ||||
| ## Configuration Examples | ||||
| 
 | ||||
| @@ -17,7 +22,7 @@ labels: | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| # Enable gzip compression | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: test-compress | ||||
| @@ -42,12 +47,6 @@ labels: | ||||
|   - "traefik.http.middlewares.test-compress.compress=true" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Enable gzip compression | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-compress.compress] | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| # Enable gzip compression | ||||
| http: | ||||
| @@ -56,24 +55,30 @@ http: | ||||
|       compress: {} | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Enable gzip compression | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-compress.compress] | ||||
| ``` | ||||
| 
 | ||||
| !!! info | ||||
|      | ||||
|     Responses are compressed when: | ||||
|      | ||||
|     * The response body is larger than `1400` bytes. | ||||
| 
 | ||||
|     Responses are compressed when the following criteria are all met: | ||||
| 
 | ||||
|     * The response body is larger than the configured minimum amount of bytes (default is `1024`). | ||||
|     * The `Accept-Encoding` request header contains `gzip`. | ||||
|     * The response is not already compressed, i.e. the `Content-Encoding` response header is not already set. | ||||
| 
 | ||||
|     If Content-Type header is not defined, or empty, the compress middleware will automatically [detect](https://mimesniff.spec.whatwg.org/) a content type.  | ||||
|     It will also set accordingly the `Content-Type` header with the detected MIME type. | ||||
|      | ||||
|     If the `Content-Type` header is not defined, or empty, the compress middleware will automatically [detect](https://mimesniff.spec.whatwg.org/) a content type. | ||||
|     It will also set the `Content-Type` header according to the detected MIME type. | ||||
| 
 | ||||
| ## Configuration Options | ||||
| 
 | ||||
| ### `excludedContentTypes` | ||||
| 
 | ||||
| `excludedContentTypes` specifies a list of content types to compare the `Content-Type` header of the incoming requests to before compressing. | ||||
| `excludedContentTypes` specifies a list of content types to compare the `Content-Type` header of the incoming requests and responses before compressing. | ||||
| 
 | ||||
| The requests with content types defined in `excludedContentTypes` are not compressed. | ||||
| The responses with content types defined in `excludedContentTypes` are not compressed. | ||||
| 
 | ||||
| Content types are compared in a case-insensitive, whitespace-ignored manner. | ||||
| 
 | ||||
| @@ -83,7 +88,7 @@ labels: | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: test-compress | ||||
| @@ -108,12 +113,6 @@ labels: | ||||
|   - "traefik.http.middlewares.test-compress.compress.excludedcontenttypes=text/event-stream" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-compress.compress] | ||||
|     excludedContentTypes = ["text/event-stream"] | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
| @@ -122,3 +121,61 @@ http: | ||||
|         excludedContentTypes: | ||||
|           - text/event-stream | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-compress.compress] | ||||
|     excludedContentTypes = ["text/event-stream"] | ||||
| ``` | ||||
| 
 | ||||
| ### `minResponseBodyBytes` | ||||
| 
 | ||||
| `minResponseBodyBytes` specifies the minimum amount of bytes a response body must have to be compressed. | ||||
| 
 | ||||
| The default value is `1024`, which should be a reasonable value for most cases. | ||||
| 
 | ||||
| Responses smaller than the specified values will not be compressed. | ||||
| 
 | ||||
| ```yaml tab="Docker" | ||||
| labels: | ||||
|   - "traefik.http.middlewares.test-compress.compress.minresponsebodybytes=1200" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: test-compress | ||||
| spec: | ||||
|   compress: | ||||
|     minResponseBodyBytes: 1200 | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Consul Catalog" | ||||
| - "traefik.http.middlewares.test-compress.compress.minresponsebodybytes=1200" | ||||
| ``` | ||||
| 
 | ||||
| ```json tab="Marathon" | ||||
| "labels": { | ||||
|   "traefik.http.middlewares.test-compress.compress.minresponsebodybytes": 1200 | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="Rancher" | ||||
| labels: | ||||
|   - "traefik.http.middlewares.test-compress.compress.minresponsebodybytes=1200" | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| http: | ||||
|   middlewares: | ||||
|     test-compress: | ||||
|       compress: | ||||
|         minResponseBodyBytes: 1200 | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| [http.middlewares] | ||||
|   [http.middlewares.test-compress.compress] | ||||
|     minResponseBodyBytes = 1200 | ||||
| ``` | ||||
| @@ -1,18 +1,23 @@ | ||||
| --- | ||||
| title: "Traefik ContentType Documentation" | ||||
| description: "Traefik Proxy's HTTP middleware can automatically specify the content-type header if it has not been defined by the backend. Read the technical documentation." | ||||
| --- | ||||
| 
 | ||||
| # ContentType | ||||
| 
 | ||||
| Handling ContentType auto-detection | ||||
| Handling Content-Type auto-detection | ||||
| {: .subtitle } | ||||
| 
 | ||||
| The Content-Type middleware - or rather its unique `autoDetect` option - | ||||
| The Content-Type middleware - or rather its `autoDetect` option - | ||||
| specifies whether to let the `Content-Type` header, | ||||
| if it has not been set by the backend, | ||||
| if it has not been defined by the backend, | ||||
| be automatically set to a value derived from the contents of the response. | ||||
| 
 | ||||
| As a proxy, the default behavior should be to leave the header alone, | ||||
| regardless of what the backend did with it. | ||||
| However, the historic default was to always auto-detect and set the header if it was nil, | ||||
| and it is going to be kept that way in order to support users currently relying on it. | ||||
| However, the historic default was to always auto-detect and set the header if it was not already defined, | ||||
| and altering this behavior would be a breaking change which would impact many users. | ||||
| 
 | ||||
| This middleware exists to enable the correct behavior until at least the default one can be changed in a future version. | ||||
| 
 | ||||
| !!! info | ||||
| @@ -21,7 +26,7 @@ This middleware exists to enable the correct behavior until at least the default | ||||
|     is still to automatically set the `Content-Type` header. | ||||
|     Therefore, given the default value of the `autoDetect` option (false), | ||||
|     simply enabling this middleware for a router switches the router's behavior. | ||||
|      | ||||
| 
 | ||||
|     The scope of the Content-Type middleware is the MIME type detection done by the core of Traefik (the server part). | ||||
|     Therefore, it has no effect against any other `Content-Type` header modifications (e.g.: in another middleware such as compress). | ||||
| 
 | ||||
| @@ -35,7 +40,7 @@ labels: | ||||
| 
 | ||||
| ```yaml tab="Kubernetes" | ||||
| # Disable auto-detection | ||||
| apiVersion: traefik.containo.us/v1alpha1 | ||||
| apiVersion: traefik.io/v1alpha1 | ||||
| kind: Middleware | ||||
| metadata: | ||||
|   name: autodetect | ||||
| @@ -61,13 +66,6 @@ labels: | ||||
|   - "traefik.http.middlewares.autodetect.contenttype.autodetect=false" | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Disable auto-detection | ||||
| [http.middlewares] | ||||
|   [http.middlewares.autodetect.contentType] | ||||
|      autoDetect=false | ||||
| ``` | ||||
| 
 | ||||
| ```yaml tab="File (YAML)" | ||||
| # Disable auto-detection | ||||
| http: | ||||
| @@ -77,6 +75,13 @@ http: | ||||
|         autoDetect: false | ||||
| ``` | ||||
| 
 | ||||
| ```toml tab="File (TOML)" | ||||
| # Disable auto-detection | ||||
| [http.middlewares] | ||||
|   [http.middlewares.autodetect.contentType] | ||||
|      autoDetect=false | ||||
| ``` | ||||
| 
 | ||||
| ## Configuration Options | ||||
| 
 | ||||
| ### `autoDetect` | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user