mirror of
				https://github.com/SquidDev-CC/CC-Tweaked
				synced 2025-11-04 07:32:59 +00:00 
			
		
		
		
	Compare commits
	
		
			1491 Commits
		
	
	
		
			v1.80pr1.8
			...
			v1.19.3-1.
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					cdc91a8e5d | ||
| 
						 | 
					8024017f53 | ||
| 
						 | 
					592ff84aea | ||
| 
						 | 
					4360458416 | ||
| 
						 | 
					717e096b94 | ||
| 
						 | 
					34a31abd9c | ||
| 
						 | 
					bdecb88cca | ||
| 
						 | 
					af15030fa4 | ||
| 
						 | 
					3a883db49e | ||
| 
						 | 
					8ea5b64f64 | ||
| 
						 | 
					7b6caf76e4 | ||
| 
						 | 
					230c7ee904 | ||
| 
						 | 
					aa203802c6 | ||
| 
						 | 
					1259e29f21 | ||
| 
						 | 
					77f62dac94 | ||
| 
						 | 
					7f34aff6bb | ||
| 
						 | 
					3047e3cdf4 | ||
| 
						 | 
					7a83a403f0 | ||
| 
						 | 
					a1d5c76d00 | ||
| 
						 | 
					bcdfa7c5ff | ||
| 
						 | 
					2c59b9122b | ||
| 
						 | 
					d2c7b944ab | ||
| 
						 | 
					e241575329 | ||
| 
						 | 
					86c4c7483d | ||
| 
						 | 
					9010219b9c | ||
| 
						 | 
					172d1824fc | ||
| 
						 | 
					9d394f44d3 | ||
| 
						 | 
					6e5b7243f4 | ||
| 
						 | 
					27b732f835 | ||
| 
						 | 
					4fa7f50534 | ||
| 
						 | 
					eeac86b07c | ||
| 
						 | 
					36ce490566 | ||
| 
						 | 
					e7fe22d4f8 | ||
| 
						 | 
					2b237332ce | ||
| 
						 | 
					1276478deb | ||
| 
						 | 
					551f6ba60c | ||
| 
						 | 
					99a2b26fc5 | ||
| 
						 | 
					0787e17ebe | ||
| 
						 | 
					06163e4f25 | ||
| 
						 | 
					18fbd96c10 | ||
| 
						 | 
					367773e173 | ||
| 
						 | 
					8007a30849 | ||
| 
						 | 
					df38f3e887 | ||
| 
						 | 
					c3fe9f00d4 | ||
| 
						 | 
					3b42f22a4f | ||
| 
						 | 
					9962ce1a5c | ||
| 
						 | 
					9f48395596 | ||
| 
						 | 
					020c5cd2d3 | ||
| 
						 | 
					a9c0b02e3c | ||
| 
						 | 
					fc5f296eeb | ||
| 
						 | 
					c96172e78d | ||
| 
						 | 
					fa122a56cf | ||
| 
						 | 
					87c6d3aef6 | ||
| 
						 | 
					95c57e843d | ||
| 
						 | 
					b13998dd96 | ||
| 
						 | 
					47816805fb | ||
| 
						 | 
					b8fce1eecc | ||
| 
						 | 
					ee2670d53b | ||
| 
						 | 
					3a96aea894 | ||
| 
						 | 
					0fc78acd49 | ||
| 
						 | 
					737d8a2585 | ||
| 
						 | 
					e2447bb0fd | ||
| 
						 | 
					2255d49d16 | ||
| 
						 | 
					3fa39b5f98 | ||
| 
						 | 
					08df68dcc0 | ||
| 
						 | 
					8f92417a2f | ||
| 
						 | 
					b58b9b7df3 | ||
| 
						 | 
					8d2e150f05 | ||
| 
						 | 
					8152f19b6e | ||
| 
						 | 
					b2b58892e3 | ||
| 
						 | 
					8360e8234d | ||
| 
						 | 
					77624fc6fd | ||
| 
						 | 
					1d335f7290 | ||
| 
						 | 
					f04acdc199 | ||
| 
						 | 
					bdf590fa30 | ||
| 
						 | 
					0c4fd2b29e | ||
| 
						 | 
					8a7156785d | ||
| 
						 | 
					4d50b48ea6 | ||
| 
						 | 
					48285404b9 | ||
| 
						 | 
					b36b96e0bc | ||
| 
						 | 
					34c7fcf750 | ||
| 
						 | 
					cc73fcd85d | ||
| 
						 | 
					22729f6f16 | ||
| 
						 | 
					55494b7671 | ||
| 
						 | 
					7d47b219c5 | ||
| 
						 | 
					320007dbc6 | ||
| 
						 | 
					0908acbe9b | ||
| 
						 | 
					e8f9cdd221 | ||
| 
						 | 
					53abe5e56e | ||
| 
						 | 
					564752c8dd | ||
| 
						 | 
					6d665ad841 | ||
| 
						 | 
					9cd728fea9 | ||
| 
						 | 
					1c890e5a5c | ||
| 
						 | 
					955b9c7d28 | ||
| 
						 | 
					76710eec9d | ||
| 
						 | 
					d8e2161f15 | ||
| 
						 | 
					c82f37d3bf | ||
| 
						 | 
					c8c128d335 | ||
| 
						 | 
					acc254a1ef | ||
| 
						 | 
					a17b001950 | ||
| 
						 | 
					e4e528e5bf | ||
| 
						 | 
					6cc86b0ae5 | ||
| 
						 | 
					f478c4ffc4 | ||
| 
						 | 
					7df0412c2d | ||
| 
						 | 
					998efcc950 | ||
| 
						 | 
					45c5de73bb | ||
| 
						 | 
					c919011a7e | ||
| 
						 | 
					0f1f5247ca | ||
| 
						 | 
					0db32bd0fe | ||
| 
						 | 
					1d3ecb551d | ||
| 
						 | 
					aefda6a381 | ||
| 
						 | 
					c1bf9f0b24 | ||
| 
						 | 
					629abb65e3 | ||
| 
						 | 
					11ac865877 | ||
| 
						 | 
					6b93fafc46 | ||
| 
						 | 
					1acb8441ec | ||
| 
						 | 
					4b0988768d | ||
| 
						 | 
					f528046535 | ||
| 
						 | 
					93f747fb54 | ||
| 
						 | 
					5d4c34fbac | ||
| 
						 | 
					4c5b3a6ee5 | ||
| 
						 | 
					7701b343fb | ||
| 
						 | 
					14cb97cba1 | ||
| 
						 | 
					1490ca8624 | ||
| 
						 | 
					f5b89982de | ||
| 
						 | 
					1a87175ae7 | ||
| 
						 | 
					c4184a33bc | ||
| 
						 | 
					b3702fed78 | ||
| 
						 | 
					b5056fc3b8 | ||
| 
						 | 
					38b2c944f3 | ||
| 
						 | 
					5ee5b11995 | ||
| 
						 | 
					b2d2153258 | ||
| 
						 | 
					3d6ef0cf96 | ||
| 
						 | 
					71f81e1201 | ||
| 
						 | 
					1e88d37004 | ||
| 
						 | 
					97387556fe | ||
| 
						 | 
					1f910ee2ba | ||
| 
						 | 
					a9ef874174 | ||
| 
						 | 
					a2911038c5 | ||
| 
						 | 
					158850be09 | ||
| 
						 | 
					be827a21db | ||
| 
						 | 
					562f224c01 | ||
| 
						 | 
					f45614175a | ||
| 
						 | 
					af7af615c7 | ||
| 
						 | 
					8171578e80 | ||
| 
						 | 
					f4e542b4db | ||
| 
						 | 
					3e3bc8d4b1 | ||
| 
						 | 
					b48f590b92 | ||
| 
						 | 
					6ab90dc30d | ||
| 
						 | 
					0cfdd7b5e9 | ||
| 
						 | 
					af5d816798 | ||
| 
						 | 
					57cf6084e2 | ||
| 
						 | 
					e9cde9e1bf | ||
| 
						 | 
					68da044ff2 | ||
| 
						 | 
					18d9993fa7 | ||
| 
						 | 
					0c3de1087e | ||
| 
						 | 
					ff89e5feeb | ||
| 
						 | 
					0b26ab366d | ||
| 
						 | 
					cb9731306c | ||
| 
						 | 
					5d833ac634 | ||
| 
						 | 
					9db3e6d2a0 | ||
| 
						 | 
					1e703f1b07 | ||
| 
						 | 
					b663028f42 | ||
| 
						 | 
					cee60cdb5b | ||
| 
						 | 
					695ef0542a | ||
| 
						 | 
					c0d20b72c9 | ||
| 
						 | 
					cf05ab1db1 | ||
| 
						 | 
					c49547b962 | ||
| 
						 | 
					c8e15f201c | ||
| 
						 | 
					bc79100a2f | ||
| 
						 | 
					9ed5ebb868 | ||
| 
						 | 
					a9b74dc979 | ||
| 
						 | 
					12b8a0393f | ||
| 
						 | 
					cbfd83c2ba | ||
| 
						 | 
					8564c1e54b | ||
| 
						 | 
					66dff1523b | ||
| 
						 | 
					08895cdecc | ||
| 
						 | 
					5be290a1e2 | ||
| 
						 | 
					371f931140 | ||
| 
						 | 
					da5956e943 | ||
| 
						 | 
					e7533f2353 | ||
| 
						 | 
					0b7fbcde53 | ||
| 
						 | 
					76f8dd2d14 | ||
| 
						 | 
					c3b7302108 | ||
| 
						 | 
					61ac48c99f | ||
| 
						 | 
					d22e138413 | ||
| 
						 | 
					ba64e06ca7 | ||
| 
						 | 
					db8c979a06 | ||
| 
						 | 
					9d18487dc5 | ||
| 
						 | 
					34a2e87735 | ||
| 
						 | 
					feb7681c9c | ||
| 
						 | 
					ad4a2aa68d | ||
| 
						 | 
					4228011b84 | ||
| 
						 | 
					c43d851e63 | ||
| 
						 | 
					50fe7935a3 | ||
| 
						 | 
					bd19fdf350 | ||
| 
						 | 
					c3615d9c5b | ||
| 
						 | 
					e2041f7438 | ||
| 
						 | 
					d61202e2b8 | ||
| 
						 | 
					6ce88a7dcf | ||
| 
						 | 
					5d65b3e654 | ||
| 
						 | 
					abf857f864 | ||
| 
						 | 
					ebef3117f2 | ||
| 
						 | 
					25a44bea6e | ||
| 
						 | 
					b28c1ac8e0 | ||
| 
						 | 
					69b211b4fb | ||
| 
						 | 
					48147fa61c | ||
| 
						 | 
					ba976f9a16 | ||
| 
						 | 
					969feb4a1c | ||
| 
						 | 
					4a273ae8e5 | ||
| 
						 | 
					bd5de11ad5 | ||
| 
						 | 
					5366fcb9c8 | ||
| 
						 | 
					6335e77da6 | ||
| 
						 | 
					4cfd0a2d1c | ||
| 
						 | 
					be3a960273 | ||
| 
						 | 
					f25a73b8f2 | ||
| 
						 | 
					954254e7e4 | ||
| 
						 | 
					e906f3ebc3 | ||
| 
						 | 
					56f0e0674f | ||
| 
						 | 
					4e438df9ad | ||
| 
						 | 
					51c3a9d8af | ||
| 
						 | 
					d967730085 | ||
| 
						 | 
					d6afee8deb | ||
| 
						 | 
					41bddcab9f | ||
| 
						 | 
					b8d7695392 | ||
| 
						 | 
					b7fa4102df | ||
| 
						 | 
					92c613a7a2 | ||
| 
						 | 
					d2f94f2653 | ||
| 
						 | 
					718111787c | ||
| 
						 | 
					bb0e449560 | ||
| 
						 | 
					ee495b3359 | ||
| 
						 | 
					d1e952770d | ||
| 
						 | 
					2d30208631 | ||
| 
						 | 
					03f50f9298 | ||
| 
						 | 
					8fc7820a12 | ||
| 
						 | 
					1a0e3fc2fa | ||
| 
						 | 
					6d5b13dbbc | ||
| 
						 | 
					f9f8233ef4 | ||
| 
						 | 
					a2e3d9d9bd | ||
| 
						 | 
					b7f698d6f7 | ||
| 
						 | 
					93f3cd4a53 | ||
| 
						 | 
					755f8eff93 | ||
| 
						 | 
					a879efc3d0 | ||
| 
						 | 
					a913232e62 | ||
| 
						 | 
					557765d8f0 | ||
| 
						 | 
					1e044aed68 | ||
| 
						 | 
					5382d34d29 | ||
| 
						 | 
					cbbb34cdd4 | ||
| 
						 | 
					8f7719a8dc | ||
| 
						 | 
					ca58e39707 | ||
| 
						 | 
					0aac966239 | ||
| 
						 | 
					0e1e8dfa8c | ||
| 
						 | 
					a1cbc1d803 | ||
| 
						 | 
					0b6dc25607 | ||
| 
						 | 
					b91809bfc7 | ||
| 
						 | 
					178126725e | ||
| 
						 | 
					cd76425877 | ||
| 
						 | 
					4411756b06 | ||
| 
						 | 
					1fd57a874f | ||
| 
						 | 
					3f0704624e | ||
| 
						 | 
					3b6cd783cb | ||
| 
						 | 
					a07bba4ece | ||
| 
						 | 
					ab22726883 | ||
| 
						 | 
					2efad38f53 | ||
| 
						 | 
					83a1af6526 | ||
| 
						 | 
					8b89d88d04 | ||
| 
						 | 
					36635662f1 | ||
| 
						 | 
					bbc0afa111 | ||
| 
						 | 
					34dc915d57 | ||
| 
						 | 
					24ed0ca723 | ||
| 
						 | 
					5052718428 | ||
| 
						 | 
					431e4c9419 | ||
| 
						 | 
					2639b84eb2 | ||
| 
						 | 
					d631111610 | ||
| 
						 | 
					c981c75b7c | ||
| 
						 | 
					f05a539443 | ||
| 
						 | 
					d8a7ab540a | ||
| 
						 | 
					a7536ea4fa | ||
| 
						 | 
					d9e75d7c47 | ||
| 
						 | 
					78334c4cb1 | ||
| 
						 | 
					f5f0c7990a | ||
| 
						 | 
					87a1c1a525 | ||
| 
						 | 
					be45b718b3 | ||
| 
						 | 
					ad2d1d6a05 | ||
| 
						 | 
					65a7370db1 | ||
| 
						 | 
					03b0244084 | ||
| 
						 | 
					6322e72110 | ||
| 
						 | 
					7ad6132494 | ||
| 
						 | 
					e2189535b8 | ||
| 
						 | 
					79467499e6 | ||
| 
						 | 
					074793090d | ||
| 
						 | 
					cbbab26bf3 | ||
| 
						 | 
					9cb7091ce7 | ||
| 
						 | 
					e909e11e05 | ||
| 
						 | 
					6239dbe9ca | ||
| 
						 | 
					49601f0b7c | ||
| 
						 | 
					caa412b7d2 | ||
| 
						 | 
					9cb7a5bec7 | ||
| 
						 | 
					118b89ea41 | ||
| 
						 | 
					f2474bbfa2 | ||
| 
						 | 
					159f90896e | ||
| 
						 | 
					f108ba93af | ||
| 
						 | 
					2a4f75ba15 | ||
| 
						 | 
					ad228e94a3 | ||
| 
						 | 
					42b98bce28 | ||
| 
						 | 
					59e3608d2a | ||
| 
						 | 
					fccca22d3f | ||
| 
						 | 
					4bfdb65989 | ||
| 
						 | 
					22e8b9b587 | ||
| 
						 | 
					77a00b14ae | ||
| 
						 | 
					78aa757549 | ||
| 
						 | 
					1196568a7c | ||
| 
						 | 
					48c4f397f9 | ||
| 
						 | 
					8871f40ced | ||
| 
						 | 
					aa62c1f206 | ||
| 
						 | 
					fd32b06d6f | ||
| 
						 | 
					739d6813c0 | ||
| 
						 | 
					daf81b897a | ||
| 
						 | 
					e865d96f7b | ||
| 
						 | 
					79b1872cab | ||
| 
						 | 
					41fa95bce4 | ||
| 
						 | 
					2a92794da3 | ||
| 
						 | 
					b3e009cca5 | ||
| 
						 | 
					ba7598c689 | ||
| 
						 | 
					70c5cbafec | ||
| 
						 | 
					2c64186965 | ||
| 
						 | 
					7731759c77 | ||
| 
						 | 
					e6339b2847 | ||
| 
						 | 
					31ba17d085 | ||
| 
						 | 
					6353e8d930 | ||
| 
						 | 
					78cce4981a | ||
| 
						 | 
					97c953a9be | ||
| 
						 | 
					52df7cb8a4 | ||
| 
						 | 
					6735cfd12e | ||
| 
						 | 
					bcc7dd6991 | ||
| 
						 | 
					f994696161 | ||
| 
						 | 
					4a4e8bb4b6 | ||
| 
						 | 
					bd36185662 | ||
| 
						 | 
					045c4fc88c | ||
| 
						 | 
					e0fcc425c6 | ||
| 
						 | 
					e01895d719 | ||
| 
						 | 
					87b38f4249 | ||
| 
						 | 
					60d1d1bb18 | ||
| 
						 | 
					cdf8b77ffd | ||
| 
						 | 
					e2ce52fe81 | ||
| 
						 | 
					9edce36efd | ||
| 
						 | 
					e05588c662 | ||
| 
						 | 
					9cf70b10ef | ||
| 
						 | 
					9ac8f3aeea | ||
| 
						 | 
					e191b08eb5 | ||
| 
						 | 
					a1221b99e1 | ||
| 
						 | 
					85bced6b1d | ||
| 
						 | 
					fc4569e0cc | ||
| 
						 | 
					e7f08313d9 | ||
| 
						 | 
					79366bf2f5 | ||
| 
						 | 
					413fa5bcc8 | ||
| 
						 | 
					79fc8237b6 | ||
| 
						 | 
					9d50d6414c | ||
| 
						 | 
					16df86224b | ||
| 
						 | 
					a9519f68a1 | ||
| 
						 | 
					f1a08a3362 | ||
| 
						 | 
					802949d888 | ||
| 
						 | 
					2b901f2d5e | ||
| 
						 | 
					62f2cd5cb2 | ||
| 
						 | 
					e558b31b2b | ||
| 
						 | 
					afd82fbf1f | ||
| 
						 | 
					901d8d4c3b | ||
| 
						 | 
					f794ce42ab | ||
| 
						 | 
					f470478a0f | ||
| 
						 | 
					aa009df740 | ||
| 
						 | 
					0c6c0badde | ||
| 
						 | 
					bed2e0b658 | ||
| 
						 | 
					0f9ddac83c | ||
| 
						 | 
					932b77d7ee | ||
| 
						 | 
					5eedea1bbb | ||
| 
						 | 
					114261944a | ||
| 
						 | 
					4d10639efb | ||
| 
						 | 
					aa36b49c50 | ||
| 
						 | 
					8a1067940d | ||
| 
						 | 
					2562642664 | ||
| 
						 | 
					632db1cfa5 | ||
| 
						 | 
					aa0d544bba | ||
| 
						 | 
					2f6ad00764 | ||
| 
						 | 
					05da4dd362 | ||
| 
						 | 
					0477b2742c | ||
| 
						 | 
					fe3c42ce22 | ||
| 
						 | 
					f6fcba7a39 | ||
| 
						 | 
					82a7edee12 | ||
| 
						 | 
					b048b6666d | ||
| 
						 | 
					e16f66e128 | ||
| 
						 | 
					1cfad31a0d | ||
| 
						 | 
					7c373c6e06 | ||
| 
						 | 
					6196aae488 | ||
| 
						 | 
					92a0ef2b75 | ||
| 
						 | 
					57c5d19f95 | ||
| 
						 | 
					1f6e0f287d | ||
| 
						 | 
					0e4b7a5a75 | ||
| 
						 | 
					47ad7a35dc | ||
| 
						 | 
					3eab2a9b57 | ||
| 
						 | 
					c4024a4c4c | ||
| 
						 | 
					f5fb82cd7d | ||
| 
						 | 
					23c17075be | ||
| 
						 | 
					e18ba8a2c2 | ||
| 
						 | 
					87988a705b | ||
| 
						 | 
					422bfdb60d | ||
| 
						 | 
					1851ed31cd | ||
| 
						 | 
					179da1d8cf | ||
| 
						 | 
					92fd93c0e0 | ||
| 
						 | 
					3929dba4a5 | ||
| 
						 | 
					af966179ce | ||
| 
						 | 
					5927e9bb10 | ||
| 
						 | 
					53811f8169 | ||
| 
						 | 
					298f339376 | ||
| 
						 | 
					2418cfb87b | ||
| 
						 | 
					9d44f1ca66 | ||
| 
						 | 
					306e06a79a | ||
| 
						 | 
					4f11549112 | ||
| 
						 | 
					7f3490591d | ||
| 
						 | 
					095101831c | ||
| 
						 | 
					7b7527ec80 | ||
| 
						 | 
					8ffd45c66e | ||
| 
						 | 
					e247bd823e | ||
| 
						 | 
					276956eed8 | ||
| 
						 | 
					a4c5ecf8df | ||
| 
						 | 
					99de00e16e | ||
| 
						 | 
					600227e481 | ||
| 
						 | 
					18d66bd727 | ||
| 
						 | 
					d3563a3854 | ||
| 
						 | 
					c2dc8bf675 | ||
| 
						 | 
					603119e1e6 | ||
| 
						 | 
					d9b3f17b52 | ||
| 
						 | 
					993bccc51f | ||
| 
						 | 
					96d3b27064 | ||
| 
						 | 
					f33f57ea35 | ||
| 
						 | 
					070479d901 | ||
| 
						 | 
					2fe40f669d | ||
| 
						 | 
					1b39c9f470 | ||
| 
						 | 
					814d5cbcd1 | ||
| 
						 | 
					4d8862c78e | ||
| 
						 | 
					6cc2f035db | ||
| 
						 | 
					cf3f1d3d48 | ||
| 
						 | 
					bca964629a | ||
| 
						 | 
					ea7a218f4a | ||
| 
						 | 
					544bcaa599 | ||
| 
						 | 
					ab6b861cd6 | ||
| 
						 | 
					72e8fc03d3 | ||
| 
						 | 
					0e94355a85 | ||
| 
						 | 
					482ae0d22e | ||
| 
						 | 
					6dd33f7099 | ||
| 
						 | 
					045472577a | ||
| 
						 | 
					9f539dbd59 | ||
| 
						 | 
					ca367e7cc7 | ||
| 
						 | 
					f6fd0ad172 | ||
| 
						 | 
					0d35331b82 | ||
| 
						 | 
					076b454c8f | ||
| 
						 | 
					36e0dcbad0 | ||
| 
						 | 
					0b5fe990e5 | ||
| 
						 | 
					29ece2a6e3 | ||
| 
						 | 
					eba26dedab | ||
| 
						 | 
					13779d6ad3 | ||
| 
						 | 
					d700f1f500 | ||
| 
						 | 
					06bf84f151 | ||
| 
						 | 
					8ba20985d7 | ||
| 
						 | 
					7bb7b5e638 | ||
| 
						 | 
					297426419b | ||
| 
						 | 
					eb61c5c5d7 | ||
| 
						 | 
					cf2bc667c1 | ||
| 
						 | 
					c8449086ee | ||
| 
						 | 
					662bead8be | ||
| 
						 | 
					acaa61a720 | ||
| 
						 | 
					5facbca2b3 | ||
| 
						 | 
					6c6b2c2ff3 | ||
| 
						 | 
					3eb601e554 | ||
| 
						 | 
					d0e79f310e | ||
| 
						 | 
					0d6528aaf0 | ||
| 
						 | 
					647902c019 | ||
| 
						 | 
					2aa70b49c1 | ||
| 
						 | 
					b17ab16e05 | ||
| 
						 | 
					b447b0e308 | ||
| 
						 | 
					94ad106272 | ||
| 
						 | 
					dc9edf26ec | ||
| 
						 | 
					048c7bda23 | ||
| 
						 | 
					c9397460a4 | ||
| 
						 | 
					b2273c9b29 | ||
| 
						 | 
					9e82209aab | ||
| 
						 | 
					340ade170f | ||
| 
						 | 
					7cac8401e8 | ||
| 
						 | 
					0f899357c2 | ||
| 
						 | 
					3396fe2871 | ||
| 
						 | 
					bbf3e48763 | ||
| 
						 | 
					92fe1d4bc2 | ||
| 
						 | 
					9fbcbae5b3 | ||
| 
						 | 
					36a779dc18 | ||
| 
						 | 
					cd8b8bbc74 | ||
| 
						 | 
					d8319bb35c | ||
| 
						 | 
					afd6adbffa | ||
| 
						 | 
					4d591c600c | ||
| 
						 | 
					0a8e427c61 | ||
| 
						 | 
					d3a5d1e314 | ||
| 
						 | 
					56010382fb | ||
| 
						 | 
					0ff6b0ca70 | ||
| 
						 | 
					4b33306940 | ||
| 
						 | 
					4dea3dff36 | ||
| 
						 | 
					3e8c741170 | ||
| 
						 | 
					62baa72457 | ||
| 
						 | 
					5eb711da87 | ||
| 
						 | 
					79c5df1d92 | ||
| 
						 | 
					991ea6e829 | ||
| 
						 | 
					1d160641a4 | ||
| 
						 | 
					c2b3d914f7 | ||
| 
						 | 
					0a537eaeee | ||
| 
						 | 
					143b2bdbcd | ||
| 
						 | 
					8cb21ed4d1 | ||
| 
						 | 
					8aa7695fdd | ||
| 
						 | 
					fa78818069 | ||
| 
						 | 
					aa857c1be3 | ||
| 
						 | 
					e4ced551eb | ||
| 
						 | 
					6eec9ba1a3 | ||
| 
						 | 
					62172c6049 | ||
| 
						 | 
					39f3cf8cbe | ||
| 
						 | 
					5082947331 | ||
| 
						 | 
					a8f675c59d | ||
| 
						 | 
					bb1ebaee4f | ||
| 
						 | 
					bb1183d274 | ||
| 
						 | 
					01ddb2b4e4 | ||
| 
						 | 
					bdd38fb061 | ||
| 
						 | 
					06f35e4997 | ||
| 
						 | 
					2d95c32892 | ||
| 
						 | 
					6f4d4540b2 | ||
| 
						 | 
					96316cddaa | ||
| 
						 | 
					e84ddef877 | ||
| 
						 | 
					11b40bb6d5 | ||
| 
						 | 
					686c6a4c44 | ||
| 
						 | 
					a1821035d3 | ||
| 
						 | 
					7b8650bbc8 | ||
| 
						 | 
					0285260e97 | ||
| 
						 | 
					10a3a223a0 | ||
| 
						 | 
					2dc970a8bb | ||
| 
						 | 
					f74c4cc83c | ||
| 
						 | 
					7012ac7163 | ||
| 
						 | 
					227b444d81 | ||
| 
						 | 
					d50db8a6f3 | ||
| 
						 | 
					3a80b51a9f | ||
| 
						 | 
					03396cf07a | ||
| 
						 | 
					5b57f7509d | ||
| 
						 | 
					0568c86628 | ||
| 
						 | 
					b31e66686d | ||
| 
						 | 
					924b8ef30f | ||
| 
						 | 
					7bcc16bb40 | ||
| 
						 | 
					31e6746bdf | ||
| 
						 | 
					c39bf3eb4d | ||
| 
						 | 
					8b952e7e1e | ||
| 
						 | 
					04e97f7b86 | ||
| 
						 | 
					74752c561c | ||
| 
						 | 
					ee96458b56 | ||
| 
						 | 
					333410e4cd | ||
| 
						 | 
					999a39a3e6 | ||
| 
						 | 
					82ca19c296 | ||
| 
						 | 
					56d8a5d585 | ||
| 
						 | 
					aa5fbb2980 | ||
| 
						 | 
					db0bb071f5 | ||
| 
						 | 
					ab702e2ba1 | ||
| 
						 | 
					d4efacd40a | ||
| 
						 | 
					347affcc5c | ||
| 
						 | 
					8ebe34b8da | ||
| 
						 | 
					7086cb8a02 | ||
| 
						 | 
					8dbc930c2f | ||
| 
						 | 
					61eb67849d | ||
| 
						 | 
					c2316ef256 | ||
| 
						 | 
					0d22270f8b | ||
| 
						 | 
					abb9c14256 | ||
| 
						 | 
					815e534dc6 | ||
| 
						 | 
					51dde077fe | ||
| 
						 | 
					31d0b7afcd | ||
| 
						 | 
					95b0d950aa | ||
| 
						 | 
					efa2be2821 | ||
| 
						 | 
					670db97fc7 | ||
| 
						 | 
					1650b72edb | ||
| 
						 | 
					a5bca3f0df | ||
| 
						 | 
					88f41314c7 | ||
| 
						 | 
					5ef8d52c13 | ||
| 
						 | 
					0b65d56ab0 | ||
| 
						 | 
					a256b70685 | ||
| 
						 | 
					f16d1499fe | ||
| 
						 | 
					79ca851e4f | ||
| 
						 | 
					d5c54d64a6 | ||
| 
						 | 
					5cfdd2339f | ||
| 
						 | 
					3ab3213290 | ||
| 
						 | 
					46c9840d00 | ||
| 
						 | 
					b3f2f14e96 | ||
| 
						 | 
					3ace49d27f | ||
| 
						 | 
					9bd662d8dc | ||
| 
						 | 
					df7a40354e | ||
| 
						 | 
					c489d4bc4f | ||
| 
						 | 
					2b029bd506 | ||
| 
						 | 
					2227845658 | ||
| 
						 | 
					2d3e88ef59 | ||
| 
						 | 
					0bfe960cbd | ||
| 
						 | 
					a735f23e1f | ||
| 
						 | 
					de6f27ceaf | ||
| 
						 | 
					2fab1a3054 | ||
| 
						 | 
					d4745ae47e | ||
| 
						 | 
					dc21e2dbc9 | ||
| 
						 | 
					75dfa71275 | ||
| 
						 | 
					d71bf225cc | ||
| 
						 | 
					8644c4ebf6 | ||
| 
						 | 
					b323db30ee | ||
| 
						 | 
					53efd6b303 | ||
| 
						 | 
					97faa1b3bc | ||
| 
						 | 
					7404133d40 | ||
| 
						 | 
					e18e24407e | ||
| 
						 | 
					026afa7f73 | ||
| 
						 | 
					29cc5bb86b | ||
| 
						 | 
					aa9d3c8269 | ||
| 
						 | 
					f8074636bc | ||
| 
						 | 
					db2cde4a4c | ||
| 
						 | 
					5eec7d9172 | ||
| 
						 | 
					8b9735d72e | ||
| 
						 | 
					1866916cb8 | ||
| 
						 | 
					f38a6a9d43 | ||
| 
						 | 
					0f6db63020 | ||
| 
						 | 
					51fcd83b87 | ||
| 
						 | 
					c2190e1318 | ||
| 
						 | 
					c40a13558c | ||
| 
						 | 
					02695aea51 | ||
| 
						 | 
					d5be1aca0e | ||
| 
						 | 
					8ff8b78ed8 | ||
| 
						 | 
					7fc55aa9a0 | ||
| 
						 | 
					38335ca187 | ||
| 
						 | 
					e0e194099c | ||
| 
						 | 
					8063059764 | ||
| 
						 | 
					f96d923b2a | ||
| 
						 | 
					9142ccfc93 | ||
| 
						 | 
					9f7cc00fcb | ||
| 
						 | 
					b129ae627b | ||
| 
						 | 
					f9fb0619fa | ||
| 
						 | 
					7f9b86a78e | ||
| 
						 | 
					58ea7a275e | ||
| 
						 | 
					8487a13764 | ||
| 
						 | 
					f0ba1108d5 | ||
| 
						 | 
					5d0daf9b2d | ||
| 
						 | 
					8b8692ba53 | ||
| 
						 | 
					1f385f5b35 | ||
| 
						 | 
					34baa09b6c | ||
| 
						 | 
					b21866fbff | ||
| 
						 | 
					e0a288bcb9 | ||
| 
						 | 
					4592534a18 | ||
| 
						 | 
					28165bfcd6 | ||
| 
						 | 
					953b94fd08 | ||
| 
						 | 
					e10e30f82b | ||
| 
						 | 
					aeb1fa0e7e | ||
| 
						 | 
					349a7543b0 | ||
| 
						 | 
					3d589eda4a | ||
| 
						 | 
					de646b66b6 | ||
| 
						 | 
					4f0d311df7 | ||
| 
						 | 
					d6e3c9a7fa | ||
| 
						 | 
					a7a724f134 | ||
| 
						 | 
					b0e30fdce1 | ||
| 
						 | 
					4e15afa254 | ||
| 
						 | 
					84bac06178 | ||
| 
						 | 
					1fecb995c9 | ||
| 
						 | 
					99b719299c | ||
| 
						 | 
					fb9590467d | ||
| 
						 | 
					bc8e090873 | ||
| 
						 | 
					cf0f67265f | ||
| 
						 | 
					53dd15a213 | ||
| 
						 | 
					eb2d617ed8 | ||
| 
						 | 
					74dae4ec17 | ||
| 
						 | 
					abbc46877b | ||
| 
						 | 
					3cb25b3525 | ||
| 
						 | 
					f387730b88 | ||
| 
						 | 
					92b45b1868 | ||
| 
						 | 
					003c7ec2e8 | ||
| 
						 | 
					c45221a2d0 | ||
| 
						 | 
					8494ba8ce2 | ||
| 
						 | 
					058d63e77f | ||
| 
						 | 
					17b5bca443 | ||
| 
						 | 
					c3f5700494 | ||
| 
						 | 
					b17ff6daf0 | ||
| 
						 | 
					e8f5531a8c | ||
| 
						 | 
					51d3b091da | ||
| 
						 | 
					9708dd6786 | ||
| 
						 | 
					e48427dbbc | ||
| 
						 | 
					669b6d2d56 | ||
| 
						 | 
					32d956bbe7 | ||
| 
						 | 
					3a147c78a8 | ||
| 
						 | 
					8c56b6a7be | ||
| 
						 | 
					66e42e0817 | ||
| 
						 | 
					0ee3d10fda | ||
| 
						 | 
					ed0afc4068 | ||
| 
						 | 
					1f70ed6985 | ||
| 
						 | 
					8f3ea60c74 | ||
| 
						 | 
					eb722a74cd | ||
| 
						 | 
					1825f67eee | ||
| 
						 | 
					975a994581 | ||
| 
						 | 
					061514549d | ||
| 
						 | 
					5e52429c23 | ||
| 
						 | 
					396cf15a1f | ||
| 
						 | 
					7514cf7320 | ||
| 
						 | 
					1316d6a3c9 | ||
| 
						 | 
					e1cbbe3628 | ||
| 
						 | 
					6d367e08a3 | ||
| 
						 | 
					eaa7359c8c | ||
| 
						 | 
					657ceda3af | ||
| 
						 | 
					a934e42219 | ||
| 
						 | 
					1544749282 | ||
| 
						 | 
					763bab80fa | ||
| 
						 | 
					417fda3019 | ||
| 
						 | 
					444830cf2d | ||
| 
						 | 
					23bf33c454 | ||
| 
						 | 
					0be030c497 | ||
| 
						 | 
					ee27d8f081 | ||
| 
						 | 
					a3a9684505 | ||
| 
						 | 
					1381325813 | ||
| 
						 | 
					52b112fae6 | ||
| 
						 | 
					c83eeb16a8 | ||
| 
						 | 
					9d1ee6f61d | ||
| 
						 | 
					b90611b4b4 | ||
| 
						 | 
					e1e7ef59c6 | ||
| 
						 | 
					9ae0f4a993 | ||
| 
						 | 
					fd262a7995 | ||
| 
						 | 
					58054ad2d1 | ||
| 
						 | 
					1255bd00fd | ||
| 
						 | 
					1f84480a80 | ||
| 
						 | 
					b838efedd2 | ||
| 
						 | 
					f78e24f9a0 | ||
| 
						 | 
					88f5b20353 | ||
| 
						 | 
					331031be45 | ||
| 
						 | 
					c5694ea966 | ||
| 
						 | 
					34b5ede326 | ||
| 
						 | 
					7b476cb24b | ||
| 
						 | 
					7ca261d763 | ||
| 
						 | 
					c864576619 | ||
| 
						 | 
					247c05305d | ||
| 
						 | 
					2232f025b8 | ||
| 
						 | 
					b2e5401486 | ||
| 
						 | 
					41226371f3 | ||
| 
						 | 
					1edb7288b9 | ||
| 
						 | 
					cc5e972cfc | ||
| 
						 | 
					92be0126df | ||
| 
						 | 
					dd6f97622e | ||
| 
						 | 
					2c9f51db89 | ||
| 
						 | 
					72340defe4 | ||
| 
						 | 
					542b66c79a | ||
| 
						 | 
					e4b0a5b3ce | ||
| 
						 | 
					f7e3e72a6e | ||
| 
						 | 
					8b17ec76a8 | ||
| 
						 | 
					b8d5a89446 | ||
| 
						 | 
					4af5bcc0b0 | ||
| 
						 | 
					96c577482d | ||
| 
						 | 
					7f9a707f75 | ||
| 
						 | 
					ed3913c1f4 | ||
| 
						 | 
					16d74dd2e8 | ||
| 
						 | 
					24bb92007a | ||
| 
						 | 
					2f0cae0bc1 | ||
| 
						 | 
					e3a672099c | ||
| 
						 | 
					abf425dfb5 | ||
| 
						 | 
					663859d2e5 | ||
| 
						 | 
					f5eb6ce03e | ||
| 
						 | 
					4ae370b9db | ||
| 
						 | 
					b97e950d86 | ||
| 
						 | 
					5865e9c41a | ||
| 
						 | 
					85cf2d5ff1 | ||
| 
						 | 
					61f8e97f6b | ||
| 
						 | 
					c92f06cfd9 | ||
| 
						 | 
					05c3c8ad32 | ||
| 
						 | 
					bb8f4c624b | ||
| 
						 | 
					ea3a160367 | ||
| 
						 | 
					737b3cb576 | ||
| 
						 | 
					d83a68f3ff | ||
| 
						 | 
					24d3777722 | ||
| 
						 | 
					826797cbd5 | ||
| 
						 | 
					511eea39a1 | ||
| 
						 | 
					24af36743d | ||
| 
						 | 
					e2761bb315 | ||
| 
						 | 
					6734a0e112 | ||
| 
						 | 
					d4199064ae | ||
| 
						 | 
					04f9644ae7 | ||
| 
						 | 
					486f41f082 | ||
| 
						 | 
					fff8353451 | ||
| 
						 | 
					9a749642d2 | ||
| 
						 | 
					c35707725f | ||
| 
						 | 
					b0651082f4 | ||
| 
						 | 
					aab0cd34cd | ||
| 
						 | 
					d2a1a00dc4 | ||
| 
						 | 
					f194f4fa3a | ||
| 
						 | 
					c9f3d315c0 | ||
| 
						 | 
					7f90f2f7ca | ||
| 
						 | 
					9f57e77ed3 | ||
| 
						 | 
					ab39cb849d | ||
| 
						 | 
					a4c9e89370 | ||
| 
						 | 
					c8aeddedd4 | ||
| 
						 | 
					83df64e520 | ||
| 
						 | 
					74ac5bb3d1 | ||
| 
						 | 
					d13bd2cce8 | ||
| 
						 | 
					ab232bd689 | ||
| 
						 | 
					cc96e41d3e | ||
| 
						 | 
					741adfa7bb | ||
| 
						 | 
					666e83cf4f | ||
| 
						 | 
					e2a635b6e5 | ||
| 
						 | 
					c58441b29c | ||
| 
						 | 
					a6fcfb6af2 | ||
| 
						 | 
					17a9329207 | ||
| 
						 | 
					f6160bdc57 | ||
| 
						 | 
					6aae4e5766 | ||
| 
						 | 
					84a6bb1cf3 | ||
| 
						 | 
					c334423d42 | ||
| 
						 | 
					113b560a20 | ||
| 
						 | 
					5bf367af9f | ||
| 
						 | 
					61fb4caaad | ||
| 
						 | 
					6734af6e4a | ||
| 
						 | 
					bf6053906d | ||
| 
						 | 
					4766833cf2 | ||
| 
						 | 
					01d81cb91d | ||
| 
						 | 
					93068402a2 | ||
| 
						 | 
					34a2c835d4 | ||
| 
						 | 
					30d35883b8 | ||
| 
						 | 
					71563a52ff | ||
| 
						 | 
					0c6e7b5db5 | ||
| 
						 | 
					334ca65482 | ||
| 
						 | 
					8472112fc1 | ||
| 
						 | 
					84036d97d9 | ||
| 
						 | 
					0832974725 | ||
| 
						 | 
					6cee4efcd3 | ||
| 
						 | 
					6f868849ab | ||
| 
						 | 
					275ca58a82 | ||
| 
						 | 
					87393e8aef | ||
| 
						 | 
					86bf57e3cd | ||
| 
						 | 
					72c1d451fe | ||
| 
						 | 
					8b4a01df27 | ||
| 
						 | 
					d0a973fa46 | ||
| 
						 | 
					748ebbe66b | ||
| 
						 | 
					59de21eae2 | ||
| 
						 | 
					50473afea8 | ||
| 
						 | 
					37f925de0a | ||
| 
						 | 
					cefde3f003 | ||
| 
						 | 
					ae6124d1f4 | ||
| 
						 | 
					7e121ff72f | ||
| 
						 | 
					5155e18de2 | ||
| 
						 | 
					7365741088 | ||
| 
						 | 
					d5368d0719 | ||
| 
						 | 
					26c12ac1a9 | ||
| 
						 | 
					2c67849b35 | ||
| 
						 | 
					04509cefec | ||
| 
						 | 
					74b9f5dcb0 | ||
| 
						 | 
					7809a2eddd | ||
| 
						 | 
					183b342071 | ||
| 
						 | 
					0bb5515055 | ||
| 
						 | 
					e8e9294fdf | ||
| 
						 | 
					9acfc0316f | ||
| 
						 | 
					29fb0baa09 | ||
| 
						 | 
					d5de39ebd4 | ||
| 
						 | 
					0faf76e4bd | ||
| 
						 | 
					99581e1f40 | ||
| 
						 | 
					e8e2ed9fe5 | ||
| 
						 | 
					9f72448ecd | ||
| 
						 | 
					3da3f16deb | ||
| 
						 | 
					0e2ce3c634 | ||
| 
						 | 
					fe00e00537 | ||
| 
						 | 
					29646a7f61 | ||
| 
						 | 
					50d2712581 | ||
| 
						 | 
					3093f882d8 | ||
| 
						 | 
					e5cf0d1c61 | ||
| 
						 | 
					cd879b067f | ||
| 
						 | 
					053cb1b53c | ||
| 
						 | 
					6b102a8142 | ||
| 
						 | 
					ac7979fb46 | ||
| 
						 | 
					c8a6888a2f | ||
| 
						 | 
					9ce33f8a3f | ||
| 
						 | 
					d51851e763 | ||
| 
						 | 
					fb70a1a998 | ||
| 
						 | 
					a1dcd59d95 | ||
| 
						 | 
					2a17585702 | ||
| 
						 | 
					2f323f23d7 | ||
| 
						 | 
					087c305b0d | ||
| 
						 | 
					31764f6d65 | ||
| 
						 | 
					4efde2b294 | ||
| 
						 | 
					95554a53d1 | ||
| 
						 | 
					89c1b2771d | ||
| 
						 | 
					8f069a9b72 | ||
| 
						 | 
					2e9d6603e3 | ||
| 
						 | 
					46595e73df | ||
| 
						 | 
					a6a1b9b8e5 | ||
| 
						 | 
					3f277a7a7b | ||
| 
						 | 
					90c5d3f1e8 | ||
| 
						 | 
					a5f7cf8334 | ||
| 
						 | 
					3075f89797 | ||
| 
						 | 
					45297665c6 | ||
| 
						 | 
					ddbf3fc111 | ||
| 
						 | 
					da82b89676 | ||
| 
						 | 
					d5f1a2c817 | ||
| 
						 | 
					6020adef6b | ||
| 
						 | 
					d2a52a8b5d | ||
| 
						 | 
					9f8774960f | ||
| 
						 | 
					36bb8b67c9 | ||
| 
						 | 
					8f3a56dd32 | ||
| 
						 | 
					113d5d982f | ||
| 
						 | 
					37a447e745 | ||
| 
						 | 
					9e2232d240 | ||
| 
						 | 
					514db30fb1 | ||
| 
						 | 
					08181f72d4 | ||
| 
						 | 
					613a28a5af | ||
| 
						 | 
					e4c422d6f9 | ||
| 
						 | 
					478f992dea | ||
| 
						 | 
					b54519d0e6 | ||
| 
						 | 
					9499654757 | ||
| 
						 | 
					c5138c535c | ||
| 
						 | 
					5bd8d84d14 | ||
| 
						 | 
					ab0310e27c | ||
| 
						 | 
					607751da40 | ||
| 
						 | 
					1efabccd14 | ||
| 
						 | 
					029374e9aa | ||
| 
						 | 
					2a8efb3fd5 | ||
| 
						 | 
					48edcde4ef | ||
| 
						 | 
					58a2995bbc | ||
| 
						 | 
					a35dcb28ef | ||
| 
						 | 
					7b2d482387 | ||
| 
						 | 
					2b077554f7 | ||
| 
						 | 
					9134f243c1 | ||
| 
						 | 
					c0f3ca81fb | ||
| 
						 | 
					190ed4fd20 | ||
| 
						 | 
					b9ff9b7f90 | ||
| 
						 | 
					b9b8121be9 | ||
| 
						 | 
					014bf55cd4 | ||
| 
						 | 
					085ae2e74a | ||
| 
						 | 
					4ff33f165d | ||
| 
						 | 
					d929c02d2a | ||
| 
						 | 
					d50a08a549 | ||
| 
						 | 
					c493d668c8 | ||
| 
						 | 
					53477fd3a1 | ||
| 
						 | 
					87aa839b60 | ||
| 
						 | 
					e02ccdcb1a | ||
| 
						 | 
					f36f532c63 | ||
| 
						 | 
					5a816917d5 | ||
| 
						 | 
					7af63d052d | ||
| 
						 | 
					4f8217d1ab | ||
| 
						 | 
					5409d441b5 | ||
| 
						 | 
					d5f82fa458 | ||
| 
						 | 
					d0deab3519 | ||
| 
						 | 
					d5a8df753a | ||
| 
						 | 
					13de2c4dd0 | ||
| 
						 | 
					906280225e | ||
| 
						 | 
					161a5b4707 | ||
| 
						 | 
					c6b6b4479c | ||
| 
						 | 
					96e7b60285 | ||
| 
						 | 
					086fccd997 | ||
| 
						 | 
					5dfaf6eee9 | ||
| 
						 | 
					e251dd066c | ||
| 
						 | 
					9abcfe56ea | ||
| 
						 | 
					abbc641fd4 | ||
| 
						 | 
					c60dcb4f5a | ||
| 
						 | 
					4be0b15afa | ||
| 
						 | 
					a4ae36b6b3 | ||
| 
						 | 
					ac075d9f53 | ||
| 
						 | 
					05d7be0362 | ||
| 
						 | 
					9a71dc1a26 | ||
| 
						 | 
					156023b154 | ||
| 
						 | 
					6b3773a862 | ||
| 
						 | 
					376d628cf0 | ||
| 
						 | 
					44062ebd52 | ||
| 
						 | 
					5739285fc2 | ||
| 
						 | 
					70b457ed18 | ||
| 
						 | 
					ca2995ed38 | ||
| 
						 | 
					6816931659 | ||
| 
						 | 
					1547ecbeb3 | ||
| 
						 | 
					e918f55b58 | ||
| 
						 | 
					c28b468844 | ||
| 
						 | 
					052cf8ee7d | ||
| 
						 | 
					550ada2f9e | ||
| 
						 | 
					17b7727262 | ||
| 
						 | 
					4553e404b2 | ||
| 
						 | 
					a565a571f9 | ||
| 
						 | 
					fb64b6017b | ||
| 
						 | 
					ed4229ab70 | ||
| 
						 | 
					3fb906ef6c | ||
| 
						 | 
					e1663f3df0 | ||
| 
						 | 
					52c6584c81 | ||
| 
						 | 
					9f87eda5de | ||
| 
						 | 
					697e9449cf | ||
| 
						 | 
					76c3e4c155 | ||
| 
						 | 
					358289b5f9 | ||
| 
						 | 
					5eec24676f | ||
| 
						 | 
					f52b8fa2de | ||
| 
						 | 
					447c3ab125 | ||
| 
						 | 
					8fac68386e | ||
| 
						 | 
					a3021c4697 | ||
| 
						 | 
					b7c61f9c6d | ||
| 
						 | 
					08a0342618 | ||
| 
						 | 
					3d7a81696d | ||
| 
						 | 
					48cb032ddf | ||
| 
						 | 
					33260a7747 | ||
| 
						 | 
					a049502d12 | ||
| 
						 | 
					ae7ef66dfa | ||
| 
						 | 
					9748679484 | ||
| 
						 | 
					da419b24e7 | ||
| 
						 | 
					7f57a977a1 | ||
| 
						 | 
					2f42a4e85b | ||
| 
						 | 
					af40f5ae5c | ||
| 
						 | 
					759d02a249 | ||
| 
						 | 
					d7729337ac | ||
| 
						 | 
					ee391ae9ea | ||
| 
						 | 
					4ed4a6409b | ||
| 
						 | 
					e5cc345f49 | ||
| 
						 | 
					d847a4d9e0 | ||
| 
						 | 
					f106733d71 | ||
| 
						 | 
					f3de97d67f | ||
| 
						 | 
					544f276ff0 | ||
| 
						 | 
					463635a459 | ||
| 
						 | 
					3b7b845930 | ||
| 
						 | 
					1fc0214857 | ||
| 
						 | 
					11bf601db9 | ||
| 
						 | 
					7c1154ddfc | ||
| 
						 | 
					df557e03fa | ||
| 
						 | 
					524b6f1d8a | ||
| 
						 | 
					cea8be7efa | ||
| 
						 | 
					c5f918ad95 | ||
| 
						 | 
					b14c7842fc | ||
| 
						 | 
					eead8b5755 | ||
| 
						 | 
					10a27a7a25 | ||
| 
						 | 
					865fc239a0 | ||
| 
						 | 
					f9f94b8304 | ||
| 
						 | 
					cb8135a0d1 | ||
| 
						 | 
					ef4b0a5632 | ||
| 
						 | 
					6a6a87489c | ||
| 
						 | 
					2360a6e951 | ||
| 
						 | 
					f4f71185ae | ||
| 
						 | 
					062977336a | ||
| 
						 | 
					e52d98ad8b | ||
| 
						 | 
					ef8da8054f | ||
| 
						 | 
					1ccd687c00 | ||
| 
						 | 
					a8ce5a5b20 | ||
| 
						 | 
					68e6bc464b | ||
| 
						 | 
					68762fe84c | ||
| 
						 | 
					419f29321a | ||
| 
						 | 
					00b41d29c1 | ||
| 
						 | 
					0ffd5fcf85 | ||
| 
						 | 
					95fee95006 | ||
| 
						 | 
					239bd769df | ||
| 
						 | 
					b4e0e9984f | ||
| 
						 | 
					79f42e35ce | ||
| 
						 | 
					be89fc25f9 | ||
| 
						 | 
					8eae02c037 | ||
| 
						 | 
					930fd59298 | ||
| 
						 | 
					bf13bac152 | ||
| 
						 | 
					649acbae1c | ||
| 
						 | 
					05eada427b | ||
| 
						 | 
					03caf9d805 | ||
| 
						 | 
					d6ea3aab1c | ||
| 
						 | 
					f3a330e330 | ||
| 
						 | 
					044d2b2b06 | ||
| 
						 | 
					fb440b0d2e | ||
| 
						 | 
					0de5969ec1 | ||
| 
						 | 
					3f98b2785e | ||
| 
						 | 
					798868427e | ||
| 
						 | 
					c79f643ba7 | ||
| 
						 | 
					1db3a14c54 | ||
| 
						 | 
					bf6d017ad1 | ||
| 
						 | 
					8b1773dd60 | ||
| 
						 | 
					a706300598 | ||
| 
						 | 
					2541c3c5e6 | ||
| 
						 | 
					41a1b99f7d | ||
| 
						 | 
					1862a439e2 | ||
| 
						 | 
					018ecfbaa0 | ||
| 
						 | 
					4c8fd4fc35 | ||
| 
						 | 
					35c1b10224 | ||
| 
						 | 
					c1c01bef7c | ||
| 
						 | 
					a48c3d0ba8 | ||
| 
						 | 
					93a9ebc4f6 | ||
| 
						 | 
					7cc2bd43c6 | ||
| 
						 | 
					393e628721 | ||
| 
						 | 
					0bcd28e58c | ||
| 
						 | 
					42f5389fb8 | ||
| 
						 | 
					041cfe91b4 | ||
| 
						 | 
					0f82a4589b | ||
| 
						 | 
					4320a4f851 | ||
| 
						 | 
					037cbabb32 | ||
| 
						 | 
					0dde859582 | ||
| 
						 | 
					e59c043fb6 | ||
| 
						 | 
					ae928c4397 | ||
| 
						 | 
					da41c65128 | ||
| 
						 | 
					4d18234714 | ||
| 
						 | 
					d254c6464b | ||
| 
						 | 
					3a5d50e572 | ||
| 
						 | 
					03b6d2f1ab | ||
| 
						 | 
					b0397ed3c5 | ||
| 
						 | 
					fa70ebcac2 | ||
| 
						 | 
					86e0330100 | ||
| 
						 | 
					92567b4d7e | ||
| 
						 | 
					0ae70fed13 | ||
| 
						 | 
					3b7300543a | ||
| 
						 | 
					642351af1a | ||
| 
						 | 
					121802a683 | ||
| 
						 | 
					08cf55e55f | ||
| 
						 | 
					3c8c0d78ef | ||
| 
						 | 
					c4d18aa9ca | ||
| 
						 | 
					2d4a87adc9 | ||
| 
						 | 
					bedac71e3d | ||
| 
						 | 
					ee4e42e730 | ||
| 
						 | 
					0de75f05dd | ||
| 
						 | 
					be6dd21e54 | ||
| 
						 | 
					927ddb0bde | ||
| 
						 | 
					a8fadabaf1 | ||
| 
						 | 
					44d0f78c1b | ||
| 
						 | 
					38f9a015ca | ||
| 
						 | 
					c311cdc6f5 | ||
| 
						 | 
					a93e0f3284 | ||
| 
						 | 
					14b3065ba4 | ||
| 
						 | 
					3ea2d6a0a8 | ||
| 
						 | 
					c802290437 | ||
| 
						 | 
					f7781defe5 | ||
| 
						 | 
					418420523a | ||
| 
						 | 
					d342a1f368 | ||
| 
						 | 
					81f85361d5 | ||
| 
						 | 
					f1621b30ec | ||
| 
						 | 
					d4f6a594b6 | ||
| 
						 | 
					ff5ba5c131 | ||
| 
						 | 
					4243f30308 | ||
| 
						 | 
					813e91073d | ||
| 
						 | 
					7250f22ff6 | ||
| 
						 | 
					db31a53bba | ||
| 
						 | 
					3023f235a4 | ||
| 
						 | 
					79cd8b4da5 | ||
| 
						 | 
					8e4d311cd9 | ||
| 
						 | 
					9bd8c86a94 | ||
| 
						 | 
					cbc0c1d0b6 | ||
| 
						 | 
					49c37857d4 | ||
| 
						 | 
					b1139a4bf6 | ||
| 
						 | 
					7e8559278e | ||
| 
						 | 
					1e7f1c98fc | ||
| 
						 | 
					a802f25dd6 | ||
| 
						 | 
					f1d6d21d6d | ||
| 
						 | 
					a80302c513 | ||
| 
						 | 
					1c46949da7 | ||
| 
						 | 
					07a56454a0 | ||
| 
						 | 
					a0e72d02c8 | ||
| 
						 | 
					455a59ca85 | ||
| 
						 | 
					46d78af068 | ||
| 
						 | 
					08d22fd3df | ||
| 
						 | 
					e6c691a8f8 | ||
| 
						 | 
					4b0e5c445c | ||
| 
						 | 
					eb5cff1045 | ||
| 
						 | 
					35c7792aa2 | ||
| 
						 | 
					521688d630 | ||
| 
						 | 
					75e2845c01 | ||
| 
						 | 
					2f96283286 | ||
| 
						 | 
					cbe6e9b5f5 | ||
| 
						 | 
					2ab79cf474 | ||
| 
						 | 
					6ce34aba79 | ||
| 
						 | 
					5eeb320b60 | ||
| 
						 | 
					93310850d2 | ||
| 
						 | 
					a2880b12ca | ||
| 
						 | 
					cef2657048 | ||
| 
						 | 
					ccd85eb055 | ||
| 
						 | 
					303b57779a | ||
| 
						 | 
					6279816ecc | ||
| 
						 | 
					4ae77261fa | ||
| 
						 | 
					4b7d843b78 | ||
| 
						 | 
					1c28df65c3 | ||
| 
						 | 
					85b740f484 | ||
| 
						 | 
					f9929cb27d | ||
| 
						 | 
					bafab1ac07 | ||
| 
						 | 
					e05c262468 | ||
| 
						 | 
					acfb72246c | ||
| 
						 | 
					9d51c4c340 | ||
| 
						 | 
					18068effec | ||
| 
						 | 
					7a3f7d3bba | ||
| 
						 | 
					95aa48c456 | ||
| 
						 | 
					6ea8ca991b | ||
| 
						 | 
					f1e551b960 | ||
| 
						 | 
					772c54ec74 | ||
| 
						 | 
					13cb789c18 | ||
| 
						 | 
					42220c4268 | ||
| 
						 | 
					3052506e2e | ||
| 
						 | 
					0741daa7eb | ||
| 
						 | 
					b4aa554279 | ||
| 
						 | 
					8fe2abe0ae | ||
| 
						 | 
					5af789ae11 | ||
| 
						 | 
					904a168d5c | ||
| 
						 | 
					724441eddc | ||
| 
						 | 
					f68ab3edd1 | ||
| 
						 | 
					68542aca3a | ||
| 
						 | 
					594bc4203c | ||
| 
						 | 
					57318b022d | ||
| 
						 | 
					761159aa93 | ||
| 
						 | 
					29dce26bf6 | ||
| 
						 | 
					717ab69093 | ||
| 
						 | 
					138a2cf08f | ||
| 
						 | 
					81daf82647 | ||
| 
						 | 
					f3798bfb63 | ||
| 
						 | 
					bc07dfad2e | ||
| 
						 | 
					8dd1c2a6cc | ||
| 
						 | 
					d10b657a54 | ||
| 
						 | 
					f90da739eb | ||
| 
						 | 
					d9cadf64e8 | ||
| 
						 | 
					15d4a55cd8 | ||
| 
						 | 
					309cbdb8be | ||
| 
						 | 
					39a9ad0ce7 | ||
| 
						 | 
					0f3c44c926 | ||
| 
						 | 
					a0e7c4a74c | ||
| 
						 | 
					7d428030df | ||
| 
						 | 
					00c395f689 | ||
| 
						 | 
					d8e1c73d26 | ||
| 
						 | 
					ffa4cc241b | ||
| 
						 | 
					6f1b740c8f | ||
| 
						 | 
					3406ba3ebf | ||
| 
						 | 
					b6715bd812 | ||
| 
						 | 
					18aee02221 | ||
| 
						 | 
					401bbf2e6a | ||
| 
						 | 
					7467b7f88a | ||
| 
						 | 
					c82d8a7c2a | ||
| 
						 | 
					6b81bcf334 | ||
| 
						 | 
					3d67421d98 | ||
| 
						 | 
					acac70675d | ||
| 
						 | 
					56434259c1 | ||
| 
						 | 
					da7e4b9016 | ||
| 
						 | 
					d4b8650d21 | ||
| 
						 | 
					17645a79f0 | ||
| 
						 | 
					ce1f14a010 | ||
| 
						 | 
					43050426de | ||
| 
						 | 
					b05f60c98b | ||
| 
						 | 
					c44c560f96 | ||
| 
						 | 
					e839ef54af | ||
| 
						 | 
					0cb659d78c | ||
| 
						 | 
					9048deeb95 | ||
| 
						 | 
					5592ebae7d | ||
| 
						 | 
					b076c32fd1 | ||
| 
						 | 
					a48f1e310f | ||
| 
						 | 
					19aca001d7 | ||
| 
						 | 
					114f913bf8 | ||
| 
						 | 
					1c9810890a | ||
| 
						 | 
					b11beb508b | ||
| 
						 | 
					af8d4da594 | ||
| 
						 | 
					a81db2cda6 | ||
| 
						 | 
					99bdff0f92 | ||
| 
						 | 
					bb138326df | ||
| 
						 | 
					5b0ce7410d | ||
| 
						 | 
					d5ea22d1a0 | ||
| 
						 | 
					210f3fa9e2 | ||
| 
						 | 
					d661cfa88b | ||
| 
						 | 
					68bf3a71dc | ||
| 
						 | 
					3cdb12d293 | ||
| 
						 | 
					ad33acd7d1 | ||
| 
						 | 
					0ec3884e98 | ||
| 
						 | 
					7f2471d6b2 | ||
| 
						 | 
					e0660b1dab | ||
| 
						 | 
					2182cfbeb7 | ||
| 
						 | 
					8fafec4915 | ||
| 
						 | 
					b9fd690ecb | ||
| 
						 | 
					2f2ada4416 | ||
| 
						 | 
					9c951c58d9 | ||
| 
						 | 
					4b4b47e231 | ||
| 
						 | 
					2c027adb68 | ||
| 
						 | 
					4a25e7a178 | ||
| 
						 | 
					55d54fec63 | ||
| 
						 | 
					220e4bd660 | ||
| 
						 | 
					978c28a686 | ||
| 
						 | 
					b867ada5e5 | ||
| 
						 | 
					7071cc972b | ||
| 
						 | 
					6898f932a0 | ||
| 
						 | 
					2e0ef6385d | ||
| 
						 | 
					f93da7ea51 | ||
| 
						 | 
					1210bb8a4d | ||
| 
						 | 
					48a71e96eb | ||
| 
						 | 
					3bf47b5290 | ||
| 
						 | 
					9e9f199e55 | ||
| 
						 | 
					5a8a111857 | ||
| 
						 | 
					48ba247ab4 | ||
| 
						 | 
					362dbd97ac | ||
| 
						 | 
					aa0e1883d1 | ||
| 
						 | 
					9cdbcb4332 | ||
| 
						 | 
					23ddd4feb5 | ||
| 
						 | 
					fcaa777c95 | ||
| 
						 | 
					b195cab6a7 | ||
| 
						 | 
					63dc0daa09 | ||
| 
						 | 
					34602ec4be | ||
| 
						 | 
					f3ce44042f | ||
| 
						 | 
					4205f18f0c | ||
| 
						 | 
					6be330ae8d | ||
| 
						 | 
					4569af2130 | ||
| 
						 | 
					765c31315a | ||
| 
						 | 
					0e191e42a0 | ||
| 
						 | 
					ca34b2a1b8 | ||
| 
						 | 
					7afc3e5260 | ||
| 
						 | 
					f9e13ca67a | ||
| 
						 | 
					810258e9b8 | ||
| 
						 | 
					5e462adc5c | ||
| 
						 | 
					1fd0b40776 | ||
| 
						 | 
					2965fb666f | ||
| 
						 | 
					390575ab4d | ||
| 
						 | 
					e4ef92ca2d | ||
| 
						 | 
					9bf586b018 | ||
| 
						 | 
					173ea72001 | ||
| 
						 | 
					1230cabcb0 | ||
| 
						 | 
					6ed03e1fcd | ||
| 
						 | 
					c4b371b124 | ||
| 
						 | 
					a600213b00 | ||
| 
						 | 
					7799b8d4cb | ||
| 
						 | 
					245bf26480 | ||
| 
						 | 
					5d05205d69 | ||
| 
						 | 
					853e2622a1 | ||
| 
						 | 
					d0bf9e9cd7 | ||
| 
						 | 
					7a7951ae68 | ||
| 
						 | 
					bd28955c8e | ||
| 
						 | 
					e46f09a939 | ||
| 
						 | 
					71b1f8138d | ||
| 
						 | 
					1d82a1c98c | ||
| 
						 | 
					b5f60f3f11 | ||
| 
						 | 
					259665d9f1 | ||
| 
						 | 
					ba823bae13 | ||
| 
						 | 
					1290a4402c | ||
| 
						 | 
					379076a5e2 | ||
| 
						 | 
					d12bdf50d8 | ||
| 
						 | 
					cbfd5aeeee | ||
| 
						 | 
					41429bdc0b | ||
| 
						 | 
					54b9966feb | ||
| 
						 | 
					105c66127c | ||
| 
						 | 
					765ad0bd3f | ||
| 
						 | 
					dd05478483 | ||
| 
						 | 
					5d028dea39 | ||
| 
						 | 
					629c51d260 | ||
| 
						 | 
					9ea57961af | ||
| 
						 | 
					07b9b1c9c7 | ||
| 
						 | 
					5b942ff9c1 | ||
| 
						 | 
					7b5a918941 | ||
| 
						 | 
					47721bf76b | ||
| 
						 | 
					35ce0974cd | ||
| 
						 | 
					52e1906d42 | ||
| 
						 | 
					eaf24a3ceb | ||
| 
						 | 
					62760e371e | ||
| 
						 | 
					e154e11186 | ||
| 
						 | 
					72d079ef61 | ||
| 
						 | 
					0bfb7049b0 | ||
| 
						 | 
					f7cb526793 | ||
| 
						 | 
					e34e833d3d | ||
| 
						 | 
					a125a19728 | ||
| 
						 | 
					b3e6a53868 | ||
| 
						 | 
					218f8e53bb | ||
| 
						 | 
					d02575528b | ||
| 
						 | 
					c78adb2cdc | ||
| 
						 | 
					3e28f79ce9 | ||
| 
						 | 
					67af7a698b | ||
| 
						 | 
					06e76f9b15 | ||
| 
						 | 
					6d383d005c | ||
| 
						 | 
					c373583723 | ||
| 
						 | 
					f1d10809d5 | ||
| 
						 | 
					474f571798 | ||
| 
						 | 
					fb9c125ab8 | ||
| 
						 | 
					162fb37421 | ||
| 
						 | 
					d953f031f0 | ||
| 
						 | 
					7fde89ad95 | ||
| 
						 | 
					bd04a93ffb | ||
| 
						 | 
					e2bfaafe28 | ||
| 
						 | 
					1fb3d16b89 | ||
| 
						 | 
					35645b3d93 | ||
| 
						 | 
					a4cd1fe77d | ||
| 
						 | 
					4145914024 | ||
| 
						 | 
					6bd11a5e4a | ||
| 
						 | 
					46fa798797 | ||
| 
						 | 
					70a226207e | ||
| 
						 | 
					257a35f3ed | ||
| 
						 | 
					af01b9514b | ||
| 
						 | 
					070fd1f2ff | ||
| 
						 | 
					fb59da2b06 | ||
| 
						 | 
					11e4d0de82 | ||
| 
						 | 
					e46ab1e267 | ||
| 
						 | 
					d6e0f368df | ||
| 
						 | 
					9f2884bc0f | ||
| 
						 | 
					18d468e887 | ||
| 
						 | 
					63f6735bb8 | ||
| 
						 | 
					ab6f0ccd16 | ||
| 
						 | 
					ae0f093e73 | ||
| 
						 | 
					e5f988e3fe | ||
| 
						 | 
					12e82afad2 | ||
| 
						 | 
					6c2db93cbd | ||
| 
						 | 
					d5edbe700b | ||
| 
						 | 
					86ad43c3ab | ||
| 
						 | 
					f450c0156b | ||
| 
						 | 
					8abcfcb4ac | ||
| 
						 | 
					f3cace1d03 | ||
| 
						 | 
					e1e5e898ab | ||
| 
						 | 
					3aa3852ff6 | ||
| 
						 | 
					709a6329c7 | ||
| 
						 | 
					c9f05a2939 | ||
| 
						 | 
					e41377f862 | ||
| 
						 | 
					d173787a94 | ||
| 
						 | 
					d5aea26f3a | ||
| 
						 | 
					2681e578c4 | ||
| 
						 | 
					1f498dcc73 | ||
| 
						 | 
					83b01d35eb | ||
| 
						 | 
					8a7e651c99 | ||
| 
						 | 
					80a5759bae | ||
| 
						 | 
					e8a4fbb4e3 | ||
| 
						 | 
					0ce67afcc1 | ||
| 
						 | 
					a8dad23fa3 | ||
| 
						 | 
					443e0f8f76 | ||
| 
						 | 
					a838595e1e | ||
| 
						 | 
					61daab910e | ||
| 
						 | 
					7fd19c43e9 | ||
| 
						 | 
					ce0685c31f | ||
| 
						 | 
					e33f852baa | ||
| 
						 | 
					227d5e9e69 | ||
| 
						 | 
					1c648850ab | ||
| 
						 | 
					63691707fc | ||
| 
						 | 
					5d97b9c8f3 | ||
| 
						 | 
					77666d7399 | ||
| 
						 | 
					ed69495b03 | ||
| 
						 | 
					66b61d4e9e | ||
| 
						 | 
					8dd084ac5c | ||
| 
						 | 
					932f8a44fc | ||
| 
						 | 
					101b3500cc | ||
| 
						 | 
					a777801e15 | ||
| 
						 | 
					0d6787641a | ||
| 
						 | 
					744bba300e | ||
| 
						 | 
					34d43d8273 | ||
| 
						 | 
					7bc9745c7a | ||
| 
						 | 
					b414cba681 | ||
| 
						 | 
					1c9110b927 | ||
| 
						 | 
					325459e336 | ||
| 
						 | 
					ee3347afbd | ||
| 
						 | 
					27aaec9a82 | ||
| 
						 | 
					929f23fd2d | ||
| 
						 | 
					8422a40c69 | ||
| 
						 | 
					ca334e7e5c | ||
| 
						 | 
					54acf1d087 | ||
| 
						 | 
					42d3901ee3 | ||
| 
						 | 
					f8b328a048 | ||
| 
						 | 
					41a320e9a4 | ||
| 
						 | 
					57fb77d7fe | ||
| 
						 | 
					b59dcbfc0e | ||
| 
						 | 
					618c534d81 | ||
| 
						 | 
					26ba61097b | ||
| 
						 | 
					2c87e66db8 | ||
| 
						 | 
					f61f7df2d8 | ||
| 
						 | 
					364d31465e | ||
| 
						 | 
					4c7ac50dd8 | ||
| 
						 | 
					37e25136ed | ||
| 
						 | 
					d9f03f3ec7 | ||
| 
						 | 
					4d5c52bc63 | ||
| 
						 | 
					a1c4a9fb58 | ||
| 
						 | 
					031f17c98e | ||
| 
						 | 
					4ead319092 | ||
| 
						 | 
					dd6bab5413 | ||
| 
						 | 
					5b48a0fa5f | ||
| 
						 | 
					2032e7a83a | ||
| 
						 | 
					72b9d3d802 | ||
| 
						 | 
					70cb8ae16c | ||
| 
						 | 
					05a97a4f12 | ||
| 
						 | 
					8b86a954ee | ||
| 
						 | 
					741ee447ca | ||
| 
						 | 
					3537f49ced | ||
| 
						 | 
					c3e4a4de5e | ||
| 
						 | 
					86569533e9 | ||
| 
						 | 
					e6850ab644 | ||
| 
						 | 
					97c2421a22 | ||
| 
						 | 
					f765b6a487 | ||
| 
						 | 
					3e6f6467af | ||
| 
						 | 
					7e6eb62504 | ||
| 
						 | 
					949b71d40c | ||
| 
						 | 
					afdfcb21b7 | ||
| 
						 | 
					776a786e1b | ||
| 
						 | 
					8ae65d1bdd | ||
| 
						 | 
					0829506176 | ||
| 
						 | 
					71ee692da0 | ||
| 
						 | 
					acd0092ed5 | ||
| 
						 | 
					1160ffbf9e | ||
| 
						 | 
					93cb6547bd | ||
| 
						 | 
					f9c91f288f | ||
| 
						 | 
					62cf921cc6 | ||
| 
						 | 
					4bd7381827 | ||
| 
						 | 
					43459ec825 | ||
| 
						 | 
					5fa01f8b96 | ||
| 
						 | 
					67d5693d2a | ||
| 
						 | 
					c2a782afa4 | ||
| 
						 | 
					14c9558ee6 | ||
| 
						 | 
					d53a73e7e7 | ||
| 
						 | 
					7e334bd4a5 | ||
| 
						 | 
					51e787f631 | ||
| 
						 | 
					8080699030 | ||
| 
						 | 
					e555f9f7f0 | ||
| 
						 | 
					822db6e9b5 | ||
| 
						 | 
					ac1f30ef43 | ||
| 
						 | 
					0fc1b8c46b | ||
| 
						 | 
					e7c19bcf55 | ||
| 
						 | 
					63ca8aca4c | ||
| 
						 | 
					2caa9c57fc | ||
| 
						 | 
					33fad2da15 | ||
| 
						 | 
					518eefbe10 | ||
| 
						 | 
					1ba73454c1 | ||
| 
						 | 
					ee4735c17c | ||
| 
						 | 
					b008edae90 | ||
| 
						 | 
					c6bd88f3ad | 
@@ -5,13 +5,25 @@ indent_style = space
 | 
			
		||||
indent_size = 4
 | 
			
		||||
end_of_line = lf
 | 
			
		||||
charset = utf-8
 | 
			
		||||
# Sadly too many files have whitespace errors, so we leave this as is for
 | 
			
		||||
# now and just make sure we don't introduce any more.
 | 
			
		||||
# trim_trailing_whitespace = true
 | 
			
		||||
trim_trailing_whitespace = true
 | 
			
		||||
insert_final_newline = true
 | 
			
		||||
 | 
			
		||||
ij_continuation_indent_size = 4
 | 
			
		||||
ij_any_do_while_brace_force = if_multiline
 | 
			
		||||
ij_any_if_brace_force = if_multiline
 | 
			
		||||
ij_any_for_brace_force = if_multiline
 | 
			
		||||
ij_any_spaces_within_array_initializer_braces = true
 | 
			
		||||
 | 
			
		||||
ij_kotlin_allow_trailing_comma = true
 | 
			
		||||
ij_kotlin_allow_trailing_comma_on_call_site = true
 | 
			
		||||
ij_kotlin_method_parameters_wrap = off
 | 
			
		||||
ij_kotlin_call_parameters_wrap = off
 | 
			
		||||
 | 
			
		||||
[*.md]
 | 
			
		||||
trim_trailing_whitespace = false
 | 
			
		||||
 | 
			
		||||
[*.properties]
 | 
			
		||||
insert_final_newline = false
 | 
			
		||||
[*.sexp]
 | 
			
		||||
indent_size = 2
 | 
			
		||||
 | 
			
		||||
[*.yml]
 | 
			
		||||
indent_size = 2
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								.git-blame-ignore-revs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								.git-blame-ignore-revs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,2 @@
 | 
			
		||||
# Reformat everything
 | 
			
		||||
f478c4ffc4fb9fc2200ec9b0bc751d047057ce81
 | 
			
		||||
							
								
								
									
										16
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,16 @@
 | 
			
		||||
# Ignore changes in generated files
 | 
			
		||||
projects/*/src/generated/** linguist-generated
 | 
			
		||||
projects/common/src/testMod/resources/data/cctest/structures/* linguist-generated
 | 
			
		||||
 | 
			
		||||
* text=auto
 | 
			
		||||
 | 
			
		||||
*.gradle eol=lf diff=java
 | 
			
		||||
*.java   eol=lf diff=java
 | 
			
		||||
*.kt     eol=lf diff=java
 | 
			
		||||
*.lua    eol=lf
 | 
			
		||||
*.md     eol=lf diff=markdown
 | 
			
		||||
*.txt    eol=lf
 | 
			
		||||
 | 
			
		||||
*.png binary
 | 
			
		||||
*.jar binary
 | 
			
		||||
*.dfpwm binary
 | 
			
		||||
							
								
								
									
										16
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							@@ -1,16 +0,0 @@
 | 
			
		||||
---
 | 
			
		||||
name: Bug report
 | 
			
		||||
about: Report some misbehaviour in the mod
 | 
			
		||||
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
<!--
 | 
			
		||||
## Before reporting
 | 
			
		||||
 - Search for the bug both here and [on the ComputerCraft issues page](https://github.com/dan200/ComputerCraft/issues?utf8=%E2%9C%93&q=is%3Aissue+)
 | 
			
		||||
 - If possible, try to reproduce on vanilla ComputerCraft. If it still occurs, [report on the ComputerCraft repo](https://github.com/dan200/ComputerCraft/issues/new) instead.
 | 
			
		||||
-->
 | 
			
		||||
 | 
			
		||||
## Useful information to include:
 | 
			
		||||
 - Minecraft version
 | 
			
		||||
 - CC: Tweaked version
 | 
			
		||||
 - Detailed reproduction steps!** Sometimes I can spot a bug pretty easily, but often it's much more obscure. Anything you can give which will help reproduce it means it'll get fixed quicker.
 | 
			
		||||
							
								
								
									
										34
									
								
								.github/ISSUE_TEMPLATE/bug_report.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								.github/ISSUE_TEMPLATE/bug_report.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,34 @@
 | 
			
		||||
name: Bug report
 | 
			
		||||
description: Report some misbehaviour in the mod
 | 
			
		||||
labels: [ bug ]
 | 
			
		||||
body:
 | 
			
		||||
- type: dropdown
 | 
			
		||||
  id: mc-version
 | 
			
		||||
  attributes:
 | 
			
		||||
    label: Minecraft Version
 | 
			
		||||
    description: What version of Minecraft are you using?
 | 
			
		||||
    options:
 | 
			
		||||
      - 1.16.x
 | 
			
		||||
      - 1.18.x
 | 
			
		||||
      - 1.19.x
 | 
			
		||||
  validations:
 | 
			
		||||
    required: true
 | 
			
		||||
- type: input
 | 
			
		||||
  id: version
 | 
			
		||||
  attributes:
 | 
			
		||||
    label: Version
 | 
			
		||||
    description: "What version of CC: Tweaked are you using?"
 | 
			
		||||
    placeholder: "e.g. 1.96.0"
 | 
			
		||||
  validations:
 | 
			
		||||
    required: true
 | 
			
		||||
- type: textarea
 | 
			
		||||
  id: details
 | 
			
		||||
  attributes:
 | 
			
		||||
    label: Details
 | 
			
		||||
    description: |
 | 
			
		||||
      Description of the bug. Please include the following:
 | 
			
		||||
      - Logs: These will be located in the `logs/` directory of your Minecraft
 | 
			
		||||
        instance. Please upload them as a gist or directly into this editor.
 | 
			
		||||
      - Detailed reproduction steps: sometimes I can spot a bug pretty easily,
 | 
			
		||||
        but often it's much more obscure. The more information I have to help
 | 
			
		||||
        reproduce it, the quicker it'll get fixed.
 | 
			
		||||
							
								
								
									
										5
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,5 @@
 | 
			
		||||
blank_issues_enabled: false
 | 
			
		||||
contact_links:
 | 
			
		||||
- name: GitHub Discussions
 | 
			
		||||
  url: https://github.com/cc-tweaked/CC-Tweaked/discussions
 | 
			
		||||
  about: Ask questions on GitHub Discussions.
 | 
			
		||||
							
								
								
									
										9
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
								
							@@ -1,15 +1,14 @@
 | 
			
		||||
---
 | 
			
		||||
name: Feature request
 | 
			
		||||
about: Suggest an idea or improvement
 | 
			
		||||
 | 
			
		||||
labels: enhancement
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
<!--
 | 
			
		||||
## Before reporting
 | 
			
		||||
 - Search for the suggestion both here and [on the ComputerCraft issues page](https://github.com/dan200/ComputerCraft/issues?utf8=%E2%9C%93&q=is%3Aissue+). It's possible someone's suggested it before!
 | 
			
		||||
 - Unless something is specific to CC:Tweaked, try to [suggest them on the ComputerCraft repo](https://github.com/dan200/ComputerCraft/issues/new). There's a lot more people watching it, so it allows the wider community to contribute.
 | 
			
		||||
 - Search for the suggestion here. It's possible someone's suggested it before!
 | 
			
		||||
-->
 | 
			
		||||
 | 
			
		||||
## Useful information to include:
 | 
			
		||||
 - Explanation of how the feature/change chould work.
 | 
			
		||||
 - Some rationale/use case for a feature. I'd like to keep CC:T as minimal
 | 
			
		||||
 - Explanation of how the feature/change should work.
 | 
			
		||||
 - Some rationale/use case for a feature. My general approach to designing new features is to ask yourself "what issue are we trying to solve" and _then_ "is this the best way to solve this issue?".
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/something_else.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/something_else.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,4 @@
 | 
			
		||||
---
 | 
			
		||||
name: Something else
 | 
			
		||||
about: An issue about something else.
 | 
			
		||||
---
 | 
			
		||||
							
								
								
									
										17
									
								
								.github/matchers/checkstyle.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								.github/matchers/checkstyle.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,17 @@
 | 
			
		||||
{
 | 
			
		||||
    "problemMatcher": [
 | 
			
		||||
        {
 | 
			
		||||
            "owner": "checkstyle",
 | 
			
		||||
            "pattern": [
 | 
			
		||||
                {
 | 
			
		||||
                    "regexp": "^([a-z]+) ([\\w./-]+):(\\d+):(\\d+): (.*)$",
 | 
			
		||||
                    "severity": 1,
 | 
			
		||||
                    "file": 2,
 | 
			
		||||
                    "line": 3,
 | 
			
		||||
                    "column": 4,
 | 
			
		||||
                    "message": 5
 | 
			
		||||
                }
 | 
			
		||||
            ]
 | 
			
		||||
        }
 | 
			
		||||
    ]
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										18
									
								
								.github/matchers/illuaminate.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.github/matchers/illuaminate.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
			
		||||
{
 | 
			
		||||
    "problemMatcher": [
 | 
			
		||||
        {
 | 
			
		||||
            "owner": "illuaminate",
 | 
			
		||||
            "severity": "warning",
 | 
			
		||||
            "pattern": [
 | 
			
		||||
                {
 | 
			
		||||
                    "regexp": "^([\\w./-]+):\\[(\\d+):(\\d+)\\-(?:\\d+):(?:\\d+)\\]: (.*) \\[([a-z:-]+)\\]$",
 | 
			
		||||
                    "file": 1,
 | 
			
		||||
                    "line": 2,
 | 
			
		||||
                    "column": 3,
 | 
			
		||||
                    "message": 4,
 | 
			
		||||
                    "code": 5
 | 
			
		||||
                }
 | 
			
		||||
            ]
 | 
			
		||||
        }
 | 
			
		||||
    ]
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										15
									
								
								.github/matchers/junit.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								.github/matchers/junit.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,15 @@
 | 
			
		||||
{
 | 
			
		||||
    "problemMatcher": [
 | 
			
		||||
        {
 | 
			
		||||
            "owner": "junit",
 | 
			
		||||
            "pattern": [
 | 
			
		||||
                {
 | 
			
		||||
                    "regexp": "^## ([\\w./-]+):(\\d+): (.*)$",
 | 
			
		||||
                    "file": 1,
 | 
			
		||||
                    "line": 2,
 | 
			
		||||
                    "message": 3
 | 
			
		||||
                }
 | 
			
		||||
            ]
 | 
			
		||||
        }
 | 
			
		||||
    ]
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										12
									
								
								.github/pull_request_template.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/pull_request_template.md
									
									
									
									
										vendored
									
									
								
							@@ -1,9 +1,3 @@
 | 
			
		||||
<!--
 | 
			
		||||
Unless this feature is specific to CC:Tweaked, try to [target the original ComputerCraft repo](https://github.com/dan200/ComputerCraft/) instead. There's a lot more people watching it, so it allows the wider community to contribute.
 | 
			
		||||
-->
 | 
			
		||||
 | 
			
		||||
## Useful information to include:
 | 
			
		||||
 - Brief explanation of the changes you've made.
 | 
			
		||||
 - Rationale of why this change has been made/reasoning behind it.
 | 
			
		||||
 | 
			
		||||
The more information you can provide, the easier it is to review something now _and_ to see why a change was made, when the code needs updating in the future.
 | 
			
		||||
## A quick checklist
 | 
			
		||||
 - If there's a existing issue, please link to it. If not, provide fill out the same information you would in a normal issue - reproduction steps for bugs, rationale for use-case.
 | 
			
		||||
 - If you're working on CraftOS, try to write a few test cases so we can ensure everything continues to work in the future. Tests live in `src/test/resources/test-rom/spec` and can be run with `./gradlew check`.
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										98
									
								
								.github/workflows/main-ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								.github/workflows/main-ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,98 @@
 | 
			
		||||
name: Build
 | 
			
		||||
 | 
			
		||||
on: [push, pull_request]
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  build:
 | 
			
		||||
    name: Build
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: Clone repository
 | 
			
		||||
      uses: actions/checkout@v3
 | 
			
		||||
 | 
			
		||||
    - name: Set up Java
 | 
			
		||||
      uses: actions/setup-java@v3
 | 
			
		||||
      with:
 | 
			
		||||
        java-version: 17
 | 
			
		||||
        distribution: 'temurin'
 | 
			
		||||
 | 
			
		||||
    - name: Setup Gradle
 | 
			
		||||
      uses: gradle/gradle-build-action@v2
 | 
			
		||||
      with:
 | 
			
		||||
        cache-read-only: ${{ !startsWith(github.ref, 'refs/heads/mc-') }}
 | 
			
		||||
 | 
			
		||||
    - name: Disable Gradle daemon
 | 
			
		||||
      run: |
 | 
			
		||||
        mkdir -p ~/.gradle
 | 
			
		||||
        echo "org.gradle.daemon=false" >> ~/.gradle/gradle.properties
 | 
			
		||||
 | 
			
		||||
    - name: Build with Gradle
 | 
			
		||||
      run: |
 | 
			
		||||
        ./gradlew assemble || ./gradlew assemble
 | 
			
		||||
        ./gradlew downloadAssets || ./gradlew downloadAssets
 | 
			
		||||
        ./gradlew build
 | 
			
		||||
 | 
			
		||||
    - name: Run client tests
 | 
			
		||||
      run: ./gradlew runGametestClient # Not checkClient, as no point running rendering tests.
 | 
			
		||||
      # These are a little flaky on GH actions: its useful to run them, but don't break the build.
 | 
			
		||||
      continue-on-error: true
 | 
			
		||||
 | 
			
		||||
    - name: Prepare Jars
 | 
			
		||||
      run: |
 | 
			
		||||
        # Find the main jar and append the git hash onto it.
 | 
			
		||||
        mkdir -p jars
 | 
			
		||||
        find projects/forge/build/libs projects/fabric/build/libs -type f -regex '.*[0-9.]+\(-SNAPSHOT\)?\.jar$' -exec bash -c 'cp {} "jars/$(basename {} .jar)-$(git rev-parse HEAD).jar"' \;
 | 
			
		||||
 | 
			
		||||
    - name: Upload Jar
 | 
			
		||||
      uses: actions/upload-artifact@v3
 | 
			
		||||
      with:
 | 
			
		||||
        name: CC-Tweaked
 | 
			
		||||
        path: ./jars
 | 
			
		||||
 | 
			
		||||
    - name: Upload coverage
 | 
			
		||||
      uses: codecov/codecov-action@v3
 | 
			
		||||
 | 
			
		||||
    - name: Parse test reports
 | 
			
		||||
      run: ./tools/parse-reports.py
 | 
			
		||||
      if: ${{ failure() }}
 | 
			
		||||
 | 
			
		||||
    - name: Run linters
 | 
			
		||||
      uses: pre-commit/action@v3.0.0
 | 
			
		||||
 | 
			
		||||
  build-core:
 | 
			
		||||
    strategy:
 | 
			
		||||
      fail-fast: false
 | 
			
		||||
      matrix:
 | 
			
		||||
        include:
 | 
			
		||||
        - name: Windows
 | 
			
		||||
          uses: windows-latest
 | 
			
		||||
 | 
			
		||||
        - name: macOS
 | 
			
		||||
          uses: macos-latest
 | 
			
		||||
 | 
			
		||||
    name: Test on ${{ matrix.name }}
 | 
			
		||||
    runs-on: ${{ matrix.uses }}
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: Clone repository
 | 
			
		||||
      uses: actions/checkout@v3
 | 
			
		||||
 | 
			
		||||
    - name: Set up Java
 | 
			
		||||
      uses: actions/setup-java@v3
 | 
			
		||||
      with:
 | 
			
		||||
        java-version: 17
 | 
			
		||||
        distribution: 'temurin'
 | 
			
		||||
 | 
			
		||||
    - name: Setup Gradle
 | 
			
		||||
      uses: gradle/gradle-build-action@v2
 | 
			
		||||
      with:
 | 
			
		||||
        cache-read-only: ${{ !startsWith(github.ref, 'refs/heads/mc-') }}
 | 
			
		||||
 | 
			
		||||
    - name: Run tests
 | 
			
		||||
      run: |
 | 
			
		||||
        ./gradlew --configure-on-demand :core:test
 | 
			
		||||
 | 
			
		||||
    - name: Parse test reports
 | 
			
		||||
      run: python3 ./tools/parse-reports.py
 | 
			
		||||
      if: ${{ failure() }}
 | 
			
		||||
							
								
								
									
										19
									
								
								.github/workflows/make-doc.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										19
									
								
								.github/workflows/make-doc.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
#!/usr/bin/env bash
 | 
			
		||||
 | 
			
		||||
set -eu
 | 
			
		||||
 | 
			
		||||
DEST="${GITHUB_REF#refs/*/}"
 | 
			
		||||
echo "Uploading docs to https://tweaked.cc/$DEST"
 | 
			
		||||
 | 
			
		||||
# Setup ssh key
 | 
			
		||||
mkdir -p "$HOME/.ssh/"
 | 
			
		||||
echo "$SSH_KEY" > "$HOME/.ssh/key"
 | 
			
		||||
chmod 600 "$HOME/.ssh/key"
 | 
			
		||||
 | 
			
		||||
# And upload
 | 
			
		||||
rsync -avc -e "ssh -i $HOME/.ssh/key -o StrictHostKeyChecking=no -p $SSH_PORT" \
 | 
			
		||||
      "$GITHUB_WORKSPACE/projects/web/build/site/" \
 | 
			
		||||
      "$SSH_USER@$SSH_HOST:/$DEST"
 | 
			
		||||
rsync -avc -e "ssh -i $HOME/.ssh/key -o StrictHostKeyChecking=no -p $SSH_PORT" \
 | 
			
		||||
      "$GITHUB_WORKSPACE/projects/common-api/build/docs/javadoc/" \
 | 
			
		||||
      "$SSH_USER@$SSH_HOST:/$DEST/javadoc"
 | 
			
		||||
							
								
								
									
										40
									
								
								.github/workflows/make-doc.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								.github/workflows/make-doc.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,40 @@
 | 
			
		||||
name: Build documentation
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  push:
 | 
			
		||||
    branches:
 | 
			
		||||
    - mc-1.19.x
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  make_doc:
 | 
			
		||||
    name: Build
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: Clone repository
 | 
			
		||||
      uses: actions/checkout@v3
 | 
			
		||||
 | 
			
		||||
    - name: Set up Java
 | 
			
		||||
      uses: actions/setup-java@v1
 | 
			
		||||
      with:
 | 
			
		||||
        java-version: 17
 | 
			
		||||
        distribution: 'temurin'
 | 
			
		||||
 | 
			
		||||
    - name: Setup Gradle
 | 
			
		||||
      uses: gradle/gradle-build-action@v2
 | 
			
		||||
      with:
 | 
			
		||||
        cache-read-only: ${{ !startsWith(github.ref, 'refs/heads/mc-') }}
 | 
			
		||||
 | 
			
		||||
    - name: Build with Gradle
 | 
			
		||||
      run: ./gradlew compileJava --no-daemon || ./gradlew compileJava --no-daemon
 | 
			
		||||
 | 
			
		||||
    - name: Generate documentation
 | 
			
		||||
      run: ./gradlew docWebsite :common-api:javadoc  --no-daemon
 | 
			
		||||
 | 
			
		||||
    - name: Upload documentation
 | 
			
		||||
      run: .github/workflows/make-doc.sh 2> /dev/null
 | 
			
		||||
      env:
 | 
			
		||||
        SSH_KEY:  ${{ secrets.SSH_KEY  }}
 | 
			
		||||
        SSH_USER: ${{ secrets.SSH_USER }}
 | 
			
		||||
        SSH_HOST: ${{ secrets.SSH_HOST }}
 | 
			
		||||
        SSH_PORT: ${{ secrets.SSH_PORT }}
 | 
			
		||||
							
								
								
									
										30
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										30
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -1,12 +1,30 @@
 | 
			
		||||
build
 | 
			
		||||
out
 | 
			
		||||
run
 | 
			
		||||
deploy
 | 
			
		||||
# Build directories
 | 
			
		||||
/classes
 | 
			
		||||
/logs
 | 
			
		||||
/build
 | 
			
		||||
/projects/*/logs
 | 
			
		||||
/projects/*/build
 | 
			
		||||
/buildSrc/build
 | 
			
		||||
/out
 | 
			
		||||
/doc/out/
 | 
			
		||||
/node_modules
 | 
			
		||||
.jqwik-database
 | 
			
		||||
 | 
			
		||||
# Runtime directories
 | 
			
		||||
/run
 | 
			
		||||
/projects/*/run
 | 
			
		||||
 | 
			
		||||
*.ipr
 | 
			
		||||
*.iws
 | 
			
		||||
*.iml
 | 
			
		||||
.idea
 | 
			
		||||
.gradle
 | 
			
		||||
luaj-2.0.3/lib
 | 
			
		||||
luaj-2.0.3/*.jar
 | 
			
		||||
*.DS_Store
 | 
			
		||||
 | 
			
		||||
/.classpath
 | 
			
		||||
/.project
 | 
			
		||||
/.settings
 | 
			
		||||
/.vscode
 | 
			
		||||
*.launch
 | 
			
		||||
 | 
			
		||||
/projects/*/src/generated/resources/.cache
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										22
									
								
								.gitpod.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								.gitpod.yml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,22 @@
 | 
			
		||||
image:
 | 
			
		||||
  file: config/gitpod/Dockerfile
 | 
			
		||||
 | 
			
		||||
ports:
 | 
			
		||||
  - port: 25565
 | 
			
		||||
    onOpen: notify
 | 
			
		||||
 | 
			
		||||
vscode:
 | 
			
		||||
  extensions:
 | 
			
		||||
    - eamodio.gitlens
 | 
			
		||||
    - github.vscode-pull-request-github
 | 
			
		||||
    - ms-azuretools.vscode-docker
 | 
			
		||||
    - redhat.java
 | 
			
		||||
    - richardwillis.vscode-gradle
 | 
			
		||||
    - vscjava.vscode-java-debug
 | 
			
		||||
    - vscode.github
 | 
			
		||||
 | 
			
		||||
tasks:
 | 
			
		||||
  - name: Setup pre-commit hool
 | 
			
		||||
    init: pre-commit install --allow-missing-config
 | 
			
		||||
  - name: Install npm packages
 | 
			
		||||
    init: npm ci
 | 
			
		||||
							
								
								
									
										54
									
								
								.pre-commit-config.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								.pre-commit-config.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,54 @@
 | 
			
		||||
# See https://pre-commit.com for more information
 | 
			
		||||
# See https://pre-commit.com/hooks.html for more hooks
 | 
			
		||||
repos:
 | 
			
		||||
- repo: https://github.com/pre-commit/pre-commit-hooks
 | 
			
		||||
  rev: v4.0.1
 | 
			
		||||
  hooks:
 | 
			
		||||
  - id: trailing-whitespace
 | 
			
		||||
  - id: end-of-file-fixer
 | 
			
		||||
  - id: check-merge-conflict
 | 
			
		||||
 | 
			
		||||
  # Quick syntax checkers
 | 
			
		||||
  - id: check-xml
 | 
			
		||||
  - id: check-yaml
 | 
			
		||||
  - id: check-toml
 | 
			
		||||
  - id: check-json
 | 
			
		||||
    exclude: "tsconfig\\.json$"
 | 
			
		||||
 | 
			
		||||
- repo: https://github.com/editorconfig-checker/editorconfig-checker.python
 | 
			
		||||
  rev: 2.3.54
 | 
			
		||||
  hooks:
 | 
			
		||||
  - id: editorconfig-checker
 | 
			
		||||
    args: ['-disable-indentation']
 | 
			
		||||
    exclude: "^(.*\\.(bat)|LICENSE)$"
 | 
			
		||||
 | 
			
		||||
- repo: local
 | 
			
		||||
  hooks:
 | 
			
		||||
  - id: license
 | 
			
		||||
    name: Spotless
 | 
			
		||||
    files: ".*\\.(java|kt|kts)$"
 | 
			
		||||
    language: system
 | 
			
		||||
    entry: ./gradlew spotlessApply
 | 
			
		||||
    pass_filenames: false
 | 
			
		||||
    require_serial: true
 | 
			
		||||
  - id: checkstyle
 | 
			
		||||
    name: Check Java codestyle
 | 
			
		||||
    files: ".*\\.java$"
 | 
			
		||||
    language: system
 | 
			
		||||
    entry: ./gradlew checkstyleMain checkstyleTest
 | 
			
		||||
    pass_filenames: false
 | 
			
		||||
    require_serial: true
 | 
			
		||||
  - id: illuaminate
 | 
			
		||||
    name: Check Lua code
 | 
			
		||||
    files: ".*\\.(lua|java|md)"
 | 
			
		||||
    language: system
 | 
			
		||||
    entry: ./gradlew lintLua
 | 
			
		||||
    pass_filenames: false
 | 
			
		||||
    require_serial: true
 | 
			
		||||
 | 
			
		||||
exclude: |
 | 
			
		||||
  (?x)^(
 | 
			
		||||
    projects/[a-z]+/src/generated|
 | 
			
		||||
    projects/core/src/test/resources/test-rom/data/json-parsing/|
 | 
			
		||||
    .*\.dfpwm
 | 
			
		||||
  )
 | 
			
		||||
							
								
								
									
										14
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								.travis.yml
									
									
									
									
									
								
							@@ -1,14 +0,0 @@
 | 
			
		||||
language: java
 | 
			
		||||
 | 
			
		||||
script: ./gradlew build --no-daemon
 | 
			
		||||
 | 
			
		||||
before_cache:
 | 
			
		||||
  - rm -f  $HOME/.gradle/caches/modules-2/modules-2.lock
 | 
			
		||||
  - rm -fr $HOME/.gradle/caches/*/plugin-resolution/
 | 
			
		||||
cache:
 | 
			
		||||
  directories:
 | 
			
		||||
    - $HOME/.gradle/caches/
 | 
			
		||||
    - $HOME/.gradle/wrapper/s
 | 
			
		||||
 | 
			
		||||
jdk:
 | 
			
		||||
    - oraclejdk8
 | 
			
		||||
							
								
								
									
										133
									
								
								CODE_OF_CONDUCT.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								CODE_OF_CONDUCT.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,133 @@
 | 
			
		||||
 | 
			
		||||
# Contributor Covenant Code of Conduct
 | 
			
		||||
 | 
			
		||||
## Our Pledge
 | 
			
		||||
 | 
			
		||||
We as members, contributors, and leaders pledge to make participation in our
 | 
			
		||||
community a harassment-free experience for everyone, regardless of age, body
 | 
			
		||||
size, visible or invisible disability, ethnicity, sex characteristics, gender
 | 
			
		||||
identity and expression, level of experience, education, socio-economic status,
 | 
			
		||||
nationality, personal appearance, race, caste, color, religion, or sexual
 | 
			
		||||
identity and orientation.
 | 
			
		||||
 | 
			
		||||
We pledge to act and interact in ways that contribute to an open, welcoming,
 | 
			
		||||
diverse, inclusive, and healthy community.
 | 
			
		||||
 | 
			
		||||
## Our Standards
 | 
			
		||||
 | 
			
		||||
Examples of behavior that contributes to a positive environment for our
 | 
			
		||||
community include:
 | 
			
		||||
 | 
			
		||||
* Demonstrating empathy and kindness toward other people
 | 
			
		||||
* Being respectful of differing opinions, viewpoints, and experiences
 | 
			
		||||
* Giving and gracefully accepting constructive feedback
 | 
			
		||||
* Accepting responsibility and apologizing to those affected by our mistakes,
 | 
			
		||||
  and learning from the experience
 | 
			
		||||
* Focusing on what is best not just for us as individuals, but for the overall
 | 
			
		||||
  community
 | 
			
		||||
 | 
			
		||||
Examples of unacceptable behavior include:
 | 
			
		||||
 | 
			
		||||
* The use of sexualized language or imagery, and sexual attention or advances of
 | 
			
		||||
  any kind
 | 
			
		||||
* Trolling, insulting or derogatory comments, and personal or political attacks
 | 
			
		||||
* Public or private harassment
 | 
			
		||||
* Publishing others' private information, such as a physical or email address,
 | 
			
		||||
  without their explicit permission
 | 
			
		||||
* Other conduct which could reasonably be considered inappropriate in a
 | 
			
		||||
  professional setting
 | 
			
		||||
 | 
			
		||||
## Enforcement Responsibilities
 | 
			
		||||
 | 
			
		||||
Community leaders are responsible for clarifying and enforcing our standards of
 | 
			
		||||
acceptable behavior and will take appropriate and fair corrective action in
 | 
			
		||||
response to any behavior that they deem inappropriate, threatening, offensive,
 | 
			
		||||
or harmful.
 | 
			
		||||
 | 
			
		||||
Community leaders have the right and responsibility to remove, edit, or reject
 | 
			
		||||
comments, commits, code, wiki edits, issues, and other contributions that are
 | 
			
		||||
not aligned to this Code of Conduct, and will communicate reasons for moderation
 | 
			
		||||
decisions when appropriate.
 | 
			
		||||
 | 
			
		||||
## Scope
 | 
			
		||||
 | 
			
		||||
This Code of Conduct applies within all community spaces, and also applies when
 | 
			
		||||
an individual is officially representing the community in public spaces.
 | 
			
		||||
Examples of representing our community include using an official e-mail address,
 | 
			
		||||
posting via an official social media account, or acting as an appointed
 | 
			
		||||
representative at an online or offline event.
 | 
			
		||||
 | 
			
		||||
## Enforcement
 | 
			
		||||
 | 
			
		||||
Instances of abusive, harassing, or otherwise unacceptable behavior may be
 | 
			
		||||
reported to the community leaders responsible for enforcement at
 | 
			
		||||
"conduct AT squiddev DOT cc". All complaints will be reviewed and investigated
 | 
			
		||||
promptly and fairly.
 | 
			
		||||
 | 
			
		||||
All community leaders are obligated to respect the privacy and security of the
 | 
			
		||||
reporter of any incident.
 | 
			
		||||
 | 
			
		||||
## Enforcement Guidelines
 | 
			
		||||
 | 
			
		||||
Community leaders will follow these Community Impact Guidelines in determining
 | 
			
		||||
the consequences for any action they deem in violation of this Code of Conduct:
 | 
			
		||||
 | 
			
		||||
### 1. Correction
 | 
			
		||||
 | 
			
		||||
**Community Impact**: Use of inappropriate language or other behavior deemed
 | 
			
		||||
unprofessional or unwelcome in the community.
 | 
			
		||||
 | 
			
		||||
**Consequence**: A private, written warning from community leaders, providing
 | 
			
		||||
clarity around the nature of the violation and an explanation of why the
 | 
			
		||||
behavior was inappropriate. A public apology may be requested.
 | 
			
		||||
 | 
			
		||||
### 2. Warning
 | 
			
		||||
 | 
			
		||||
**Community Impact**: A violation through a single incident or series of
 | 
			
		||||
actions.
 | 
			
		||||
 | 
			
		||||
**Consequence**: A warning with consequences for continued behavior. No
 | 
			
		||||
interaction with the people involved, including unsolicited interaction with
 | 
			
		||||
those enforcing the Code of Conduct, for a specified period of time. This
 | 
			
		||||
includes avoiding interactions in community spaces as well as external channels
 | 
			
		||||
like social media. Violating these terms may lead to a temporary or permanent
 | 
			
		||||
ban.
 | 
			
		||||
 | 
			
		||||
### 3. Temporary Ban
 | 
			
		||||
 | 
			
		||||
**Community Impact**: A serious violation of community standards, including
 | 
			
		||||
sustained inappropriate behavior.
 | 
			
		||||
 | 
			
		||||
**Consequence**: A temporary ban from any sort of interaction or public
 | 
			
		||||
communication with the community for a specified period of time. No public or
 | 
			
		||||
private interaction with the people involved, including unsolicited interaction
 | 
			
		||||
with those enforcing the Code of Conduct, is allowed during this period.
 | 
			
		||||
Violating these terms may lead to a permanent ban.
 | 
			
		||||
 | 
			
		||||
### 4. Permanent Ban
 | 
			
		||||
 | 
			
		||||
**Community Impact**: Demonstrating a pattern of violation of community
 | 
			
		||||
standards, including sustained inappropriate behavior, harassment of an
 | 
			
		||||
individual, or aggression toward or disparagement of classes of individuals.
 | 
			
		||||
 | 
			
		||||
**Consequence**: A permanent ban from any sort of public interaction within the
 | 
			
		||||
community.
 | 
			
		||||
 | 
			
		||||
## Attribution
 | 
			
		||||
 | 
			
		||||
This Code of Conduct is adapted from the [Contributor Covenant][homepage],
 | 
			
		||||
version 2.1, available at
 | 
			
		||||
[https://www.contributor-covenant.org/version/2/1/code_of_conduct.html][v2.1].
 | 
			
		||||
 | 
			
		||||
Community Impact Guidelines were inspired by
 | 
			
		||||
[Mozilla's code of conduct enforcement ladder][Mozilla CoC].
 | 
			
		||||
 | 
			
		||||
For answers to common questions about this code of conduct, see the FAQ at
 | 
			
		||||
[https://www.contributor-covenant.org/faq][FAQ]. Translations are available at
 | 
			
		||||
[https://www.contributor-covenant.org/translations][translations].
 | 
			
		||||
 | 
			
		||||
[homepage]: https://www.contributor-covenant.org
 | 
			
		||||
[v2.1]: https://www.contributor-covenant.org/version/2/1/code_of_conduct.html
 | 
			
		||||
[Mozilla CoC]: https://github.com/mozilla/diversity
 | 
			
		||||
[FAQ]: https://www.contributor-covenant.org/faq
 | 
			
		||||
[translations]: https://www.contributor-covenant.org/translations
 | 
			
		||||
							
								
								
									
										105
									
								
								CONTRIBUTING.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								CONTRIBUTING.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,105 @@
 | 
			
		||||
# Contributing to CC: Tweaked
 | 
			
		||||
As with many open source projects, CC: Tweaked thrives on contributions from other people! This document (hopefully)
 | 
			
		||||
provides an introduction as to how to get started in helping out.
 | 
			
		||||
 | 
			
		||||
If you've any other questions, [just ask the community][community] or [open an issue][new-issue].
 | 
			
		||||
 | 
			
		||||
## Table of Contents
 | 
			
		||||
 - [Reporting issues](#reporting-issues)
 | 
			
		||||
 - [Translations](#translations)
 | 
			
		||||
 - [Setting up a development environment](#setting-up-a-development-environment)
 | 
			
		||||
 - [Developing CC: Tweaked](#developing-cc-tweaked)
 | 
			
		||||
 - [Writing documentation](#writing-documentation)
 | 
			
		||||
 | 
			
		||||
## Reporting issues
 | 
			
		||||
If you have a bug, suggestion, or other feedback, the best thing to do is [file an issue][new-issue]. When doing so, do
 | 
			
		||||
use the issue templates - they provide a useful hint on what information to provide.
 | 
			
		||||
 | 
			
		||||
## Translations
 | 
			
		||||
Translations are managed through [Weblate], an online interface for managing language strings. This is synced
 | 
			
		||||
automatically with GitHub, so please don't submit PRs adding/changing translations!
 | 
			
		||||
 | 
			
		||||
## Setting up a development environment
 | 
			
		||||
In order to develop CC: Tweaked, you'll need to download the source code and then run it.
 | 
			
		||||
 | 
			
		||||
 - Make sure you've got the following software instealled:
 | 
			
		||||
   - Java Development Kit (JDK) installed. This can be downloaded from [Adoptium].
 | 
			
		||||
   - [Git](https://git-scm.com/).
 | 
			
		||||
   - If you want to work on documentation, [NodeJS][node].
 | 
			
		||||
 | 
			
		||||
 - Download CC: Tweaked's source code:
 | 
			
		||||
   ```
 | 
			
		||||
   git clone https://github.com/cc-tweaked/CC-Tweaked.git
 | 
			
		||||
   cd CC-Tweaked
 | 
			
		||||
   ```
 | 
			
		||||
 | 
			
		||||
 - Build CC: Tweaked with `./gradlew build`. This will be very slow the first time it runs, as it needs to download a
 | 
			
		||||
   lot of dependencies (and decompile Minecraft several times). Subsequent runs should be much faster!
 | 
			
		||||
 | 
			
		||||
 - You're now ready to start developing CC: Tweaked. Running `./gradlew :forge:runClient` or
 | 
			
		||||
   `./gradle :fabric:runClient` will start Minecraft under Forge and Fabric respectively.
 | 
			
		||||
 | 
			
		||||
If you want to run CC:T in a normal Minecraft instance, run `./gradlew assemble` and copy the `.jar` from
 | 
			
		||||
`projects/forge/build/libs` (for Forge) or `projects/fabric/build/libs` (for Fabric).
 | 
			
		||||
 | 
			
		||||
## Developing CC: Tweaked
 | 
			
		||||
Before making any major changes to CC: Tweaked, I'd recommend you have a read of the [the architecture
 | 
			
		||||
document][architecture] first. While it's not a comprehensive document, it gives a good hint of where you should start
 | 
			
		||||
looking to make your changes. As always, if you're not sure [do ask the community][community]!
 | 
			
		||||
 | 
			
		||||
### Testing
 | 
			
		||||
When making larger changes, it's may be useful to write a test to make sure your code works as expected.
 | 
			
		||||
 | 
			
		||||
CC: Tweaked has several test suites, each designed to test something different:
 | 
			
		||||
 | 
			
		||||
 - In order to test CraftOS and its builtin APIs, we have a test suite written in Lua located at
 | 
			
		||||
   `projects/core/src/test/resources/test-rom/`. These don't rely on any Minecraft code, which means they can run on
 | 
			
		||||
   emulators, acting as a sort of compliance test.
 | 
			
		||||
 | 
			
		||||
   These tests are written using a test system called "mcfly", heavily inspired by [busted]. Groups of tests go inside
 | 
			
		||||
   `describe` blocks, and a single test goes inside `it`. Assertions are generally written using `expect` (inspired by
 | 
			
		||||
   Hamcrest and the like). For instance, `expect(foo):eq("bar")` asserts that your variable `foo` is equal to the
 | 
			
		||||
   expected value `"bar"`.
 | 
			
		||||
 | 
			
		||||
   These tests can be run with `./gradlew :core:test`.
 | 
			
		||||
 | 
			
		||||
 - In-game functionality, such as the behaviour of blocks and items, is tested using [Minecraft's gametest
 | 
			
		||||
   system][mc-test] (`projects/common/src/testMod`). These tests spin up a server, spawn a structure for each test, and
 | 
			
		||||
   then run some code on the blocks defined in that structure.
 | 
			
		||||
 | 
			
		||||
   These tests can be run with `./gradlew runGametest` (or `./gradle :forge:runGametest`/`./gradlew :fabric:runGametest`
 | 
			
		||||
   for a single loader).
 | 
			
		||||
 | 
			
		||||
For more information, [see the architecture document][architecture].
 | 
			
		||||
 | 
			
		||||
## Writing documentation
 | 
			
		||||
When writing documentation for [CC: Tweaked's documentation website][docs], it may be useful to build the documentation
 | 
			
		||||
and preview it yourself before submitting a PR.
 | 
			
		||||
 | 
			
		||||
You'll first need to [set up a development environment as above](#setting-up-a-development-environment).
 | 
			
		||||
 | 
			
		||||
Once this is set up, you can now run `./gradlew docWebsite`. This generates documentation from our Lua and Java code,
 | 
			
		||||
writing the resulting HTML into `./projects/web/build/site`, which can then be opened in a browser. When iterating on
 | 
			
		||||
documentation, you can instead run `./gradlew docWebsite -t`, which will rebuild documentation every time you change a
 | 
			
		||||
file.
 | 
			
		||||
 | 
			
		||||
Documentation is built using [illuaminate] which, while not currently documented (somewhat ironic), is largely the same
 | 
			
		||||
as [ldoc][ldoc]. Documentation comments are written in Markdown, though note that we do not support many GitHub-specific
 | 
			
		||||
markdown features - if you can, do check what the documentation looks like locally!
 | 
			
		||||
 | 
			
		||||
When writing long-form documentation (such as the guides in [doc/guides](doc/guides)), I find it useful to tell a
 | 
			
		||||
narrative. Think of what you want the user to learn or achieve, then start introducing a simple concept and then talk
 | 
			
		||||
about how you can build on that, until you've covered everything!
 | 
			
		||||
 | 
			
		||||
[new-issue]: https://github.com/cc-tweaked/CC-Tweaked/issues/new/choose "Create a new issue"
 | 
			
		||||
[community]: README.md#community "Get in touch with the community."
 | 
			
		||||
[Adoptium]: https://adoptium.net/temurin/releases?version=17 "Download OpenJDK 17"
 | 
			
		||||
[checkstyle]: https://checkstyle.org/
 | 
			
		||||
[illuaminate]: https://github.com/SquidDev/illuaminate/ "Illuaminate on GitHub"
 | 
			
		||||
[weblate]: https://i18n.tweaked.cc/projects/cc-tweaked/minecraft/ "CC: Tweaked weblate instance"
 | 
			
		||||
[docs]: https://tweaked.cc/ "CC: Tweaked documentation"
 | 
			
		||||
[ldoc]: http://stevedonovan.github.io/ldoc/ "ldoc, a Lua documentation generator."
 | 
			
		||||
[mc-test]: https://www.youtube.com/watch?v=vXaWOJTCYNg
 | 
			
		||||
[busted]: https://github.com/Olivine-Labs/busted "busted: Elegant Lua unit testing."
 | 
			
		||||
[node]: https://nodejs.org/en/ "Node.js"
 | 
			
		||||
[architecture]: projects/ARCHITECTURE.md
 | 
			
		||||
							
								
								
									
										19
									
								
								LICENSE-luaj
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								LICENSE-luaj
									
									
									
									
									
								
							@@ -1,19 +0,0 @@
 | 
			
		||||
Copyright (c) 2007 LuaJ. All rights reserved.
 | 
			
		||||
 | 
			
		||||
Permission is hereby granted, free of charge, to any person obtaining a copy
 | 
			
		||||
of this software and associated documentation files (the "Software"), to deal
 | 
			
		||||
in the Software without restriction, including without limitation the rights
 | 
			
		||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | 
			
		||||
copies of the Software, and to permit persons to whom the Software is
 | 
			
		||||
furnished to do so, subject to the following conditions:
 | 
			
		||||
 | 
			
		||||
The above copyright notice and this permission notice shall be included in
 | 
			
		||||
all copies or substantial portions of the Software.
 | 
			
		||||
 | 
			
		||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
			
		||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
			
		||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | 
			
		||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
			
		||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | 
			
		||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 | 
			
		||||
THE SOFTWARE.
 | 
			
		||||
							
								
								
									
										99
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										99
									
								
								README.md
									
									
									
									
									
								
							@@ -1,47 +1,66 @@
 | 
			
		||||
# 
 | 
			
		||||
[](https://travis-ci.org/SquidDev-CC/CC-Tweaked)
 | 
			
		||||
# 
 | 
			
		||||
[](https://github.com/cc-tweaked/CC-Tweaked/actions "Current build status") [][CurseForge]
 | 
			
		||||
 | 
			
		||||
CC: Tweaked is a fork of ComputerCraft which aims to provide earlier access to the more experimental and in-development
 | 
			
		||||
features of the mod. For a more stable experience, I recommend checking out the
 | 
			
		||||
[original mod](https://github.com/dan200/ComputerCraft).
 | 
			
		||||
CC: Tweaked is a mod for Minecraft which adds programmable computers, turtles and more to the game. A fork of the
 | 
			
		||||
much-beloved [ComputerCraft], it continues its legacy with better performance, stability, and a wealth of new features.
 | 
			
		||||
 | 
			
		||||
## What?
 | 
			
		||||
CC: Tweaked (or CC:T for short) does not aim to create a competing fork of ComputerCraft, nor am I planning to take it
 | 
			
		||||
in in a vastly different direction to the original mod. In fact, CC:T aims to be a nurturing ground for various
 | 
			
		||||
features, with a pull request against the original mod being the end goal.
 | 
			
		||||
 | 
			
		||||
CC:T also includes many pull requests from the community which have not yet been merged, offering a large number
 | 
			
		||||
of additional bug fixes and features over the original mod.
 | 
			
		||||
 | 
			
		||||
## Features
 | 
			
		||||
CC: Tweaked contains the all features of the latest alpha, as well as numerous fixes, performance improvements and
 | 
			
		||||
several additional features. I'd recommend checking out [the releases page](https://github.com/SquidDev-CC/CC-Tweaked/releases)
 | 
			
		||||
to see the full changes, but here's a couple of the more interesting changes:
 | 
			
		||||
 | 
			
		||||
 - Replace LuaJ with Cobalt.
 | 
			
		||||
 - Allow running multiple computers at the same time.
 | 
			
		||||
 - Websocket support in the HTTP library.
 | 
			
		||||
 - Wired modems and cables act more like multiparts.
 | 
			
		||||
 - Add map-like rendering for pocket computers and printed pages/books.
 | 
			
		||||
 - Adds the `/computercraft` command, offering various diagnostic tools for server owners. This allows operators to
 | 
			
		||||
   track which computers are hogging resources, turn on and shutdown multiple computers at once and interact with
 | 
			
		||||
   computers remotely.
 | 
			
		||||
 - Add full-block wired modems, allowing one to wrap non-solid peripherals (such as turtles, or chests if Plethora is
 | 
			
		||||
   installed).
 | 
			
		||||
 | 
			
		||||
## Relation to CCTweaks?
 | 
			
		||||
This mod has nothing to do with CCTweaks, though there is no denying the name is a throwback to it. That being said,
 | 
			
		||||
several features have been included, such as full block modems, the Cobalt runtime and map-like rendering for pocket
 | 
			
		||||
computers.
 | 
			
		||||
CC: Tweaked can be installed from [CurseForge] or [Modrinth]. It requires the [Minecraft Forge][forge] mod loader, but
 | 
			
		||||
[versions are available for Fabric][ccrestitched].
 | 
			
		||||
 | 
			
		||||
## Contributing
 | 
			
		||||
Any contribution is welcome, be that using the mod, reporting bugs or contributing code. If you do wish to contribute
 | 
			
		||||
code, do consider submitting it to the ComputerCraft repository instead.
 | 
			
		||||
Any contribution is welcome, be that using the mod, reporting bugs or contributing code. If you want to get started
 | 
			
		||||
developing the mod, [check out the instructions here](CONTRIBUTING.md#developing).
 | 
			
		||||
 | 
			
		||||
That being said, in order to start helping develop CC:T, you'll need to follow these steps:
 | 
			
		||||
## Community
 | 
			
		||||
If you need help getting started with CC: Tweaked, want to show off your latest project, or just want to chat about
 | 
			
		||||
ComputerCraft, do check out our [forum] and [GitHub discussions page][GitHub discussions]! There's also a fairly
 | 
			
		||||
populated, albeit quiet [IRC channel][irc], if that's more your cup of tea.
 | 
			
		||||
 | 
			
		||||
 - **Clone the repository:** `git clone https://github.com/SquidDev-CC/CC-Tweaked.git && cd CC-Tweaked`
 | 
			
		||||
 - **Setup Forge:** `./gradlew setupDecompWorkspace`
 | 
			
		||||
 - **Test your changes:** `./gradlew runClient` (or run the `GradleStart` class from your IDE).
 | 
			
		||||
We also host fairly comprehensive documentation at [tweaked.cc](https://tweaked.cc/ "The CC: Tweaked website").
 | 
			
		||||
 | 
			
		||||
If you want to run CC:T in a normal Minecraft instance, run `./gradlew build` and copy the `.jar` from `build/libs`.
 | 
			
		||||
## Using
 | 
			
		||||
CC: Tweaked is hosted on my maven repo, and so is relatively simple to depend on. You may wish to add a soft (or hard)
 | 
			
		||||
dependency in your `mods.toml` file, with the appropriate version bounds, to ensure that API functionality you depend
 | 
			
		||||
on is present.
 | 
			
		||||
 | 
			
		||||
```groovy
 | 
			
		||||
repositories {
 | 
			
		||||
  maven {
 | 
			
		||||
    url "https://squiddev.cc/maven/"
 | 
			
		||||
    content {
 | 
			
		||||
      includeGroup("cc.tweaked")
 | 
			
		||||
      includeModule("org.squiddev", "Cobalt")
 | 
			
		||||
      includeModule("fuzs.forgeconfigapiport", "forgeconfigapiport-fabric")
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
  // Vanilla (i.e. for multi-loader systems)
 | 
			
		||||
  compileOnly("cc.tweaked:cc-tweaked-$mcVersion-common-api")
 | 
			
		||||
 | 
			
		||||
  // Forge Gradle
 | 
			
		||||
  compileOnly(fg.deobf("cc.tweaked:cc-tweaked-$mcVersion-forge-api:$cctVersion"))
 | 
			
		||||
  runtimeOnly(fg.deobf("cc.tweaked:cc-tweaked-$mcVersion-forge:$cctVersion"))
 | 
			
		||||
 | 
			
		||||
  // Fabric Loom
 | 
			
		||||
  modCompileOnly("cc.tweaked:cc-tweaked-$mcVersion-fabric-api:$cctVersion")
 | 
			
		||||
  modRuntimeOnly("cc.tweaked:cc-tweaked-$mcVersion-fabric:$cctVersion")
 | 
			
		||||
}
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
You should also be careful to only use classes within the `dan200.computercraft.api` package. Non-API classes are
 | 
			
		||||
subject to change at any point. If you depend on functionality outside the API, file an issue, and we can look into
 | 
			
		||||
exposing more features.
 | 
			
		||||
 | 
			
		||||
We bundle the API sources with the jar, so documentation should be easily viewable within your editor. Alternatively,
 | 
			
		||||
the generated documentation [can be browsed online](https://tweaked.cc/javadoc/).
 | 
			
		||||
 | 
			
		||||
[computercraft]: https://github.com/dan200/ComputerCraft "ComputerCraft on GitHub"
 | 
			
		||||
[curseforge]: https://minecraft.curseforge.com/projects/cc-tweaked "Download CC: Tweaked from CurseForge"
 | 
			
		||||
[modrinth]: https://modrinth.com/mod/gu7yAYhd "Download CC: Tweaked from Modrinth"
 | 
			
		||||
[forge]: https://files.minecraftforge.net/ "Download Minecraft Forge."
 | 
			
		||||
[ccrestitched]: https://www.curseforge.com/minecraft/mc-mods/cc-restitched "Download CC: Restitched from CurseForge"
 | 
			
		||||
[forum]: https://forums.computercraft.cc/
 | 
			
		||||
[GitHub Discussions]: https://github.com/cc-tweaked/CC-Tweaked/discussions
 | 
			
		||||
[IRC]: https://webchat.esper.net/?channels=computercraft "#computercraft on EsperNet"
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										198
									
								
								build.gradle
									
									
									
									
									
								
							
							
						
						
									
										198
									
								
								build.gradle
									
									
									
									
									
								
							@@ -1,198 +0,0 @@
 | 
			
		||||
 | 
			
		||||
// For those who want the bleeding edge
 | 
			
		||||
buildscript {
 | 
			
		||||
    repositories {
 | 
			
		||||
        jcenter()
 | 
			
		||||
        maven {
 | 
			
		||||
            name = "forge"
 | 
			
		||||
            url = "http://files.minecraftforge.net/maven"
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    dependencies {
 | 
			
		||||
        classpath 'net.minecraftforge.gradle:ForgeGradle:2.3-SNAPSHOT'
 | 
			
		||||
        classpath 'org.ajoberstar:gradle-git:1.6.0'
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
plugins {
 | 
			
		||||
    id 'com.matthewprenger.cursegradle' version '1.0.10'
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
apply plugin: 'net.minecraftforge.gradle.forge'
 | 
			
		||||
apply plugin: 'org.ajoberstar.grgit'
 | 
			
		||||
apply plugin: 'maven-publish'
 | 
			
		||||
apply plugin: 'maven'
 | 
			
		||||
 | 
			
		||||
version = "1.80pr1.8"
 | 
			
		||||
group = "org.squiddev"
 | 
			
		||||
archivesBaseName = "cc-tweaked"
 | 
			
		||||
 | 
			
		||||
minecraft {
 | 
			
		||||
    version = "1.12.2-14.23.4.2749"
 | 
			
		||||
    runDir = "run"
 | 
			
		||||
    replace '${version}', project.version
 | 
			
		||||
 | 
			
		||||
    // the mappings can be changed at any time, and must be in the following format.
 | 
			
		||||
    // snapshot_YYYYMMDD   snapshot are built nightly.
 | 
			
		||||
    // stable_#            stables are built at the discretion of the MCP team.
 | 
			
		||||
    // Use non-default mappings at your own risk. they may not allways work.
 | 
			
		||||
    // simply re-run your setup task after changing the mappings to update your workspace.
 | 
			
		||||
    mappings = "snapshot_20180724"
 | 
			
		||||
    // makeObfSourceJar = false // an Srg named sources jar is made by default. uncomment this to disable.
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
repositories {
 | 
			
		||||
    maven {
 | 
			
		||||
        name = "JEI"
 | 
			
		||||
        url  = "http://dvs1.progwml6.com/files/maven"
 | 
			
		||||
    }
 | 
			
		||||
    maven {
 | 
			
		||||
        name = "squiddev"
 | 
			
		||||
        url = "https://dl.bintray.com/squiddev/maven"
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    ivy { artifactPattern "https://asie.pl/files/mods/Charset/LibOnly/[module]-[revision](-[classifier]).[ext]" }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
configurations {
 | 
			
		||||
    shade
 | 
			
		||||
    compile.extendsFrom shade
 | 
			
		||||
    deployerJars
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
    deobfProvided "mezz.jei:jei_1.12.2:4.8.5.159:api"
 | 
			
		||||
    deobfProvided "pl.asie:Charset-Lib:0.5.4.6"
 | 
			
		||||
 | 
			
		||||
    runtime "mezz.jei:jei_1.12.2:4.8.5.159"
 | 
			
		||||
 | 
			
		||||
    shade 'org.squiddev:Cobalt:0.3.2'
 | 
			
		||||
 | 
			
		||||
    testCompile 'junit:junit:4.11'
 | 
			
		||||
 | 
			
		||||
    deployerJars "org.apache.maven.wagon:wagon-ssh:3.0.0"
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
javadoc {
 | 
			
		||||
    include "dan200/computercraft/api/**/*.java"
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
jar {
 | 
			
		||||
    dependsOn javadoc
 | 
			
		||||
 | 
			
		||||
    manifest {
 | 
			
		||||
        attributes('FMLAT': 'computercraft_at.cfg')
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    into("docs", { from (javadoc.destinationDir) })
 | 
			
		||||
 | 
			
		||||
    into("api", { from (sourceSets.main.allSource) {
 | 
			
		||||
        include "dan200/computercraft/api/**/*.java"
 | 
			
		||||
    }})
 | 
			
		||||
 | 
			
		||||
    from configurations.shade.collect { it.isDirectory() ? it : zipTree(it) }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
import org.ajoberstar.grgit.Grgit
 | 
			
		||||
 | 
			
		||||
processResources {
 | 
			
		||||
    inputs.property "version", project.version
 | 
			
		||||
    inputs.property "mcversion", project.minecraft.version
 | 
			
		||||
 | 
			
		||||
    def grgit = Grgit.open(dir: '.')
 | 
			
		||||
    inputs.property "commithash", grgit.head().id
 | 
			
		||||
 | 
			
		||||
    def blacklist = ['GitHub', 'dan200', 'Daniel Ratcliffe']
 | 
			
		||||
    Set<String> contributors = []
 | 
			
		||||
 | 
			
		||||
    grgit.log().each {
 | 
			
		||||
        if (!blacklist.contains(it.author.name)) contributors.add(it.author.name)
 | 
			
		||||
        if (!blacklist.contains(it.committer.name)) contributors.add(it.committer.name)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    from(sourceSets.main.resources.srcDirs) {
 | 
			
		||||
        include 'mcmod.info'
 | 
			
		||||
        include 'assets/computercraft/lua/rom/help/credits.txt'
 | 
			
		||||
 | 
			
		||||
        expand 'version':project.version,
 | 
			
		||||
               'mcversion':project.minecraft.version,
 | 
			
		||||
               'gitcontributors':contributors.sort(false, String.CASE_INSENSITIVE_ORDER).join('\n')
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    from(sourceSets.main.resources.srcDirs) {
 | 
			
		||||
        exclude 'mcmod.info'
 | 
			
		||||
        exclude 'assets/computercraft/lua/rom/help/credits.txt'
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
curseforge {
 | 
			
		||||
    apiKey = project.hasProperty('curseForgeApiKey') ? project.curseForgeApiKey : ''
 | 
			
		||||
    project {
 | 
			
		||||
        id = '282001'
 | 
			
		||||
        releaseType = 'beta'
 | 
			
		||||
        changelog = ''
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
publishing {
 | 
			
		||||
    publications {
 | 
			
		||||
        mavenJava(MavenPublication) {
 | 
			
		||||
            from components.java
 | 
			
		||||
            artifact sourceJar
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uploadArchives {
 | 
			
		||||
    repositories {
 | 
			
		||||
        if(project.hasProperty('mavenUploadUrl')) {
 | 
			
		||||
            mavenDeployer {
 | 
			
		||||
                configuration = configurations.deployerJars
 | 
			
		||||
 | 
			
		||||
                repository(url: project.property('mavenUploadUrl')) {
 | 
			
		||||
                    authentication(
 | 
			
		||||
                        userName: project.property('mavenUploadUser'),
 | 
			
		||||
                        privateKey: project.property('mavenUploadKey'))
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                pom.project {
 | 
			
		||||
                    name 'CC: Tweaked'
 | 
			
		||||
                    packaging 'jar'
 | 
			
		||||
                    description 'A fork of ComputerCraft which aims to provide earlier access to the more experimental and in-development features of the mod.'
 | 
			
		||||
                    url 'https://github.com/SquidDev-CC/CC-Tweaked'
 | 
			
		||||
 | 
			
		||||
                    scm {
 | 
			
		||||
                        url 'https://github.com/dan200/ComputerCraft.git'
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    issueManagement {
 | 
			
		||||
                        system 'github'
 | 
			
		||||
                        url 'https://github.com/dan200/ComputerCraft/issues'
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    licenses {
 | 
			
		||||
                        license {
 | 
			
		||||
                            name 'ComputerCraft Public License, Version 1.0'
 | 
			
		||||
                            url 'https://github.com/dan200/ComputerCraft/blob/master/LICENSE'
 | 
			
		||||
                            distribution 'repo'
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                pom.whenConfigured { pom ->
 | 
			
		||||
                    pom.dependencies.clear()
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
gradle.projectsEvaluated {
 | 
			
		||||
    tasks.withType(JavaCompile) {
 | 
			
		||||
        options.compilerArgs << "-Xlint"
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
runClient.outputs.upToDateWhen { false }
 | 
			
		||||
runServer.outputs.upToDateWhen { false }
 | 
			
		||||
							
								
								
									
										52
									
								
								build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,52 @@
 | 
			
		||||
import org.jetbrains.gradle.ext.compiler
 | 
			
		||||
import org.jetbrains.gradle.ext.settings
 | 
			
		||||
 | 
			
		||||
plugins {
 | 
			
		||||
    publishing
 | 
			
		||||
    alias(libs.plugins.taskTree)
 | 
			
		||||
    alias(libs.plugins.githubRelease)
 | 
			
		||||
    id("org.jetbrains.gradle.plugin.idea-ext")
 | 
			
		||||
    id("cc-tweaked")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
val isUnstable = project.properties["isUnstable"] == "true"
 | 
			
		||||
val modVersion: String by extra
 | 
			
		||||
val mcVersion: String by extra
 | 
			
		||||
 | 
			
		||||
githubRelease {
 | 
			
		||||
    token(findProperty("githubApiKey") as String? ?: "")
 | 
			
		||||
    owner.set("cc-tweaked")
 | 
			
		||||
    repo.set("CC-Tweaked")
 | 
			
		||||
    targetCommitish.set(cct.gitBranch)
 | 
			
		||||
 | 
			
		||||
    tagName.set("v$mcVersion-$modVersion")
 | 
			
		||||
    releaseName.set("[$mcVersion] $modVersion")
 | 
			
		||||
    body.set(
 | 
			
		||||
        provider {
 | 
			
		||||
            "## " + project(":core").file("src/main/resources/data/computercraft/lua/rom/help/whatsnew.md")
 | 
			
		||||
                .readLines()
 | 
			
		||||
                .takeWhile { it != "Type \"help changelog\" to see the full version history." }
 | 
			
		||||
                .joinToString("\n").trim()
 | 
			
		||||
        },
 | 
			
		||||
    )
 | 
			
		||||
    prerelease.set(isUnstable)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.publish { dependsOn(tasks.githubRelease) }
 | 
			
		||||
 | 
			
		||||
idea.project.settings.compiler.javac {
 | 
			
		||||
    // We want ErrorProne to be present when compiling via IntelliJ, as it offers some helpful warnings
 | 
			
		||||
    // and errors. Loop through our source sets and find the appropriate flags.
 | 
			
		||||
    moduleJavacAdditionalOptions = subprojects
 | 
			
		||||
        .asSequence()
 | 
			
		||||
        .map { evaluationDependsOn(it.path) }
 | 
			
		||||
        .flatMap { project ->
 | 
			
		||||
            val sourceSets = project.extensions.findByType(SourceSetContainer::class) ?: return@flatMap sequenceOf()
 | 
			
		||||
            sourceSets.asSequence().map { sourceSet ->
 | 
			
		||||
                val name = "${idea.project.name}.${project.name}.${sourceSet.name}"
 | 
			
		||||
                val compile = project.tasks.named(sourceSet.compileJavaTaskName, JavaCompile::class).get()
 | 
			
		||||
                name to compile.options.allCompilerArgs.joinToString(" ") { if (it.contains(" ")) "\"$it\"" else it }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        .toMap()
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										70
									
								
								buildSrc/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								buildSrc/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,70 @@
 | 
			
		||||
plugins {
 | 
			
		||||
    `java-gradle-plugin`
 | 
			
		||||
    `kotlin-dsl`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Duplicated in settings.gradle.kts
 | 
			
		||||
repositories {
 | 
			
		||||
    mavenCentral()
 | 
			
		||||
    gradlePluginPortal()
 | 
			
		||||
 | 
			
		||||
    maven("https://maven.minecraftforge.net") {
 | 
			
		||||
        name = "Forge"
 | 
			
		||||
        content {
 | 
			
		||||
            includeGroup("net.minecraftforge")
 | 
			
		||||
            includeGroup("net.minecraftforge.gradle")
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    maven("https://maven.parchmentmc.org") {
 | 
			
		||||
        name = "Librarian"
 | 
			
		||||
        content {
 | 
			
		||||
            includeGroupByRegex("^org\\.parchmentmc.*")
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    maven("https://repo.spongepowered.org/repository/maven-public/") {
 | 
			
		||||
        name = "Sponge"
 | 
			
		||||
        content {
 | 
			
		||||
            includeGroup("org.spongepowered")
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    maven("https://maven.fabricmc.net/") {
 | 
			
		||||
        name = "Fabric"
 | 
			
		||||
        content {
 | 
			
		||||
            includeGroup("net.fabricmc")
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
    implementation(libs.errorProne.plugin)
 | 
			
		||||
    implementation(libs.kotlin.plugin)
 | 
			
		||||
    implementation(libs.spotless)
 | 
			
		||||
 | 
			
		||||
    implementation(libs.fabric.loom)
 | 
			
		||||
    implementation(libs.forgeGradle)
 | 
			
		||||
    implementation(libs.librarian)
 | 
			
		||||
    implementation(libs.quiltflower)
 | 
			
		||||
    implementation(libs.vanillaGradle)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
gradlePlugin {
 | 
			
		||||
    plugins {
 | 
			
		||||
        register("cc-tweaked") {
 | 
			
		||||
            id = "cc-tweaked"
 | 
			
		||||
            implementationClass = "cc.tweaked.gradle.CCTweakedPlugin"
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        register("cc-tweaked.illuaminate") {
 | 
			
		||||
            id = "cc-tweaked.illuaminate"
 | 
			
		||||
            implementationClass = "cc.tweaked.gradle.IlluaminatePlugin"
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        register("cc-tweaked.node") {
 | 
			
		||||
            id = "cc-tweaked.node"
 | 
			
		||||
            implementationClass = "cc.tweaked.gradle.NodePlugin"
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										7
									
								
								buildSrc/settings.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								buildSrc/settings.gradle.kts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,7 @@
 | 
			
		||||
dependencyResolutionManagement {
 | 
			
		||||
    versionCatalogs {
 | 
			
		||||
        create("libs") {
 | 
			
		||||
            from(files("../gradle/libs.versions.toml"))
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										66
									
								
								buildSrc/src/main/kotlin/cc-tweaked.fabric.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								buildSrc/src/main/kotlin/cc-tweaked.fabric.gradle.kts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,66 @@
 | 
			
		||||
/** Default configuration for Fabric projects. */
 | 
			
		||||
 | 
			
		||||
import cc.tweaked.gradle.CCTweakedExtension
 | 
			
		||||
import cc.tweaked.gradle.CCTweakedPlugin
 | 
			
		||||
import cc.tweaked.gradle.IdeaRunConfigurations
 | 
			
		||||
import cc.tweaked.gradle.MinecraftConfigurations
 | 
			
		||||
 | 
			
		||||
plugins {
 | 
			
		||||
    `java-library`
 | 
			
		||||
    id("fabric-loom")
 | 
			
		||||
    id("io.github.juuxel.loom-quiltflower")
 | 
			
		||||
    id("cc-tweaked.java-convention")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
plugins.apply(CCTweakedPlugin::class.java)
 | 
			
		||||
 | 
			
		||||
val mcVersion: String by extra
 | 
			
		||||
 | 
			
		||||
repositories {
 | 
			
		||||
    maven("https://maven.parchmentmc.org/") {
 | 
			
		||||
        name = "Parchment"
 | 
			
		||||
        content {
 | 
			
		||||
            includeGroup("org.parchmentmc.data")
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
loom {
 | 
			
		||||
    splitEnvironmentSourceSets()
 | 
			
		||||
    splitModDependencies.set(true)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
MinecraftConfigurations.setup(project)
 | 
			
		||||
 | 
			
		||||
extensions.configure(CCTweakedExtension::class.java) {
 | 
			
		||||
    linters(minecraft = true, loader = "fabric")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
    val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs")
 | 
			
		||||
 | 
			
		||||
    minecraft("com.mojang:minecraft:$mcVersion")
 | 
			
		||||
    mappings(
 | 
			
		||||
        loom.layered {
 | 
			
		||||
            officialMojangMappings()
 | 
			
		||||
            parchment(
 | 
			
		||||
                project.dependencies.create(
 | 
			
		||||
                    group = "org.parchmentmc.data",
 | 
			
		||||
                    name = "parchment-${libs.findVersion("parchmentMc").get()}",
 | 
			
		||||
                    version = libs.findVersion("parchment").get().toString(),
 | 
			
		||||
                    ext = "zip",
 | 
			
		||||
                ),
 | 
			
		||||
            )
 | 
			
		||||
        },
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    modImplementation(libs.findLibrary("fabric-loader").get())
 | 
			
		||||
    modImplementation(libs.findLibrary("fabric-api").get())
 | 
			
		||||
 | 
			
		||||
    // Depend on error prone annotations to silence a lot of compile warnings.
 | 
			
		||||
    compileOnlyApi(libs.findLibrary("errorProne.annotations").get())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.ideaSyncTask {
 | 
			
		||||
    doLast { IdeaRunConfigurations(project).patch() }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										39
									
								
								buildSrc/src/main/kotlin/cc-tweaked.forge.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								buildSrc/src/main/kotlin/cc-tweaked.forge.gradle.kts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
			
		||||
/** Default configuration for Forge projects. */
 | 
			
		||||
 | 
			
		||||
import cc.tweaked.gradle.CCTweakedExtension
 | 
			
		||||
import cc.tweaked.gradle.CCTweakedPlugin
 | 
			
		||||
import cc.tweaked.gradle.IdeaRunConfigurations
 | 
			
		||||
import cc.tweaked.gradle.MinecraftConfigurations
 | 
			
		||||
 | 
			
		||||
plugins {
 | 
			
		||||
    id("cc-tweaked.java-convention")
 | 
			
		||||
    id("net.minecraftforge.gradle")
 | 
			
		||||
    id("org.parchmentmc.librarian.forgegradle")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
plugins.apply(CCTweakedPlugin::class.java)
 | 
			
		||||
 | 
			
		||||
val mcVersion: String by extra
 | 
			
		||||
 | 
			
		||||
minecraft {
 | 
			
		||||
    val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs")
 | 
			
		||||
    mappings("parchment", "${libs.findVersion("parchmentMc").get()}-${libs.findVersion("parchment").get()}-$mcVersion")
 | 
			
		||||
 | 
			
		||||
    accessTransformer(project(":forge").file("src/main/resources/META-INF/accesstransformer.cfg"))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
MinecraftConfigurations.setup(project)
 | 
			
		||||
 | 
			
		||||
extensions.configure(CCTweakedExtension::class.java) {
 | 
			
		||||
    linters(minecraft = true, loader = "forge")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
    val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs")
 | 
			
		||||
    "minecraft"("net.minecraftforge:forge:$mcVersion-${libs.findVersion("forge").get()}")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.configureEach {
 | 
			
		||||
    // genIntellijRuns isn't registered until much later, so we need this silly hijinks.
 | 
			
		||||
    if (name == "genIntellijRuns") doLast { IdeaRunConfigurations(project).patch() }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										60
									
								
								buildSrc/src/main/kotlin/cc-tweaked.gametest.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								buildSrc/src/main/kotlin/cc-tweaked.gametest.gradle.kts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,60 @@
 | 
			
		||||
import cc.tweaked.gradle.clientClasses
 | 
			
		||||
import cc.tweaked.gradle.commonClasses
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Sets up the configurations for writing game tests.
 | 
			
		||||
 *
 | 
			
		||||
 * See notes in [cc.tweaked.gradle.MinecraftConfigurations] for the general design behind these cursed ideas.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
plugins {
 | 
			
		||||
    id("cc-tweaked.kotlin-convention")
 | 
			
		||||
    id("cc-tweaked.java-convention")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
val main = sourceSets["main"]
 | 
			
		||||
val client = sourceSets["client"]
 | 
			
		||||
 | 
			
		||||
// Both testMod and testFixtures inherit from the main and client classpath, just so we have access to Minecraft classes.
 | 
			
		||||
val testMod by sourceSets.creating {
 | 
			
		||||
    compileClasspath += main.compileClasspath + client.compileClasspath
 | 
			
		||||
    runtimeClasspath += main.runtimeClasspath + client.runtimeClasspath
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
configurations {
 | 
			
		||||
    named(testMod.compileClasspathConfigurationName) {
 | 
			
		||||
        shouldResolveConsistentlyWith(compileClasspath.get())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    named(testMod.runtimeClasspathConfigurationName) {
 | 
			
		||||
        shouldResolveConsistentlyWith(runtimeClasspath.get())
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Like the main test configurations, we're safe to depend on source set outputs.
 | 
			
		||||
dependencies {
 | 
			
		||||
    add(testMod.implementationConfigurationName, main.output)
 | 
			
		||||
    add(testMod.implementationConfigurationName, client.output)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Similar to java-test-fixtures, but tries to avoid putting the obfuscated jar on the classpath.
 | 
			
		||||
 | 
			
		||||
val testFixtures by sourceSets.creating {
 | 
			
		||||
    compileClasspath += main.compileClasspath + client.compileClasspath
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
java.registerFeature("testFixtures") {
 | 
			
		||||
    usingSourceSet(testFixtures)
 | 
			
		||||
    disablePublication()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
    val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs")
 | 
			
		||||
    add(testFixtures.apiConfigurationName, libs.findBundle("test").get())
 | 
			
		||||
    // Consumers of this project already have the common and client classes on the classpath, so it's fine for these
 | 
			
		||||
    // to be compile-only.
 | 
			
		||||
    add(testFixtures.compileOnlyApiConfigurationName, commonClasses(project))
 | 
			
		||||
    add(testFixtures.compileOnlyApiConfigurationName, clientClasses(project))
 | 
			
		||||
 | 
			
		||||
    testImplementation(testFixtures(project))
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										198
									
								
								buildSrc/src/main/kotlin/cc-tweaked.java-convention.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										198
									
								
								buildSrc/src/main/kotlin/cc-tweaked.java-convention.gradle.kts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,198 @@
 | 
			
		||||
import cc.tweaked.gradle.CCTweakedExtension
 | 
			
		||||
import cc.tweaked.gradle.CCTweakedPlugin
 | 
			
		||||
import cc.tweaked.gradle.LicenseHeader
 | 
			
		||||
import com.diffplug.gradle.spotless.FormatExtension
 | 
			
		||||
import com.diffplug.spotless.LineEnding
 | 
			
		||||
import net.ltgt.gradle.errorprone.CheckSeverity
 | 
			
		||||
import net.ltgt.gradle.errorprone.errorprone
 | 
			
		||||
import java.nio.charset.StandardCharsets
 | 
			
		||||
 | 
			
		||||
plugins {
 | 
			
		||||
    `java-library`
 | 
			
		||||
    idea
 | 
			
		||||
    jacoco
 | 
			
		||||
    checkstyle
 | 
			
		||||
    id("com.diffplug.spotless")
 | 
			
		||||
    id("net.ltgt.errorprone")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
val modVersion: String by extra
 | 
			
		||||
val mcVersion: String by extra
 | 
			
		||||
 | 
			
		||||
group = "cc.tweaked"
 | 
			
		||||
version = modVersion
 | 
			
		||||
 | 
			
		||||
base.archivesName.convention("cc-tweaked-$mcVersion-${project.name}")
 | 
			
		||||
 | 
			
		||||
java {
 | 
			
		||||
    toolchain {
 | 
			
		||||
        languageVersion.set(CCTweakedPlugin.JAVA_VERSION)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    withSourcesJar()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
repositories {
 | 
			
		||||
    mavenCentral()
 | 
			
		||||
    maven("https://squiddev.cc/maven") {
 | 
			
		||||
        name = "SquidDev"
 | 
			
		||||
        content {
 | 
			
		||||
            includeGroup("org.squiddev")
 | 
			
		||||
            includeGroup("cc.tweaked")
 | 
			
		||||
            // Things we mirror
 | 
			
		||||
            includeGroup("dev.architectury")
 | 
			
		||||
            includeGroup("maven.modrinth")
 | 
			
		||||
            includeGroup("me.shedaniel")
 | 
			
		||||
            includeGroup("me.shedaniel.cloth")
 | 
			
		||||
            includeGroup("mezz.jei")
 | 
			
		||||
            includeModule("com.terraformersmc", "modmenu")
 | 
			
		||||
            includeModule("fuzs.forgeconfigapiport", "forgeconfigapiport-fabric")
 | 
			
		||||
            // Until https://github.com/SpongePowered/Mixin/pull/593 is merged
 | 
			
		||||
            includeModule("org.spongepowered", "mixin")
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
    val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs")
 | 
			
		||||
    checkstyle(libs.findLibrary("checkstyle").get())
 | 
			
		||||
 | 
			
		||||
    errorprone(libs.findLibrary("errorProne-core").get())
 | 
			
		||||
    errorprone(libs.findLibrary("nullAway").get())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Configure default JavaCompile tasks with our arguments.
 | 
			
		||||
sourceSets.all {
 | 
			
		||||
    tasks.named(compileJavaTaskName, JavaCompile::class.java) {
 | 
			
		||||
        // Processing just gives us "No processor claimed any of these annotations", so skip that!
 | 
			
		||||
        options.compilerArgs.addAll(listOf("-Xlint", "-Xlint:-processing"))
 | 
			
		||||
 | 
			
		||||
        options.errorprone {
 | 
			
		||||
            check("InvalidBlockTag", CheckSeverity.OFF) // Broken by @cc.xyz
 | 
			
		||||
            check("InvalidParam", CheckSeverity.OFF) // Broken by records.
 | 
			
		||||
            check("InlineMeSuggester", CheckSeverity.OFF) // Minecraft uses @Deprecated liberally
 | 
			
		||||
            // Too many false positives right now. Maybe we need an indirection for it later on.
 | 
			
		||||
            check("ReferenceEquality", CheckSeverity.OFF)
 | 
			
		||||
            check("UnusedVariable", CheckSeverity.OFF) // Too many false positives with records.
 | 
			
		||||
            check("OperatorPrecedence", CheckSeverity.OFF) // For now.
 | 
			
		||||
            check("AlreadyChecked", CheckSeverity.OFF) // Seems to be broken?
 | 
			
		||||
            check("NonOverridingEquals", CheckSeverity.OFF) // Peripheral.equals makes this hard to avoid
 | 
			
		||||
            check("FutureReturnValueIgnored", CheckSeverity.OFF) // Too many false positives with Netty
 | 
			
		||||
 | 
			
		||||
            check("NullAway", CheckSeverity.ERROR)
 | 
			
		||||
            option("NullAway:AnnotatedPackages", listOf("dan200.computercraft", "net.fabricmc.fabric.api").joinToString(","))
 | 
			
		||||
            option("NullAway:ExcludedFieldAnnotations", listOf("org.spongepowered.asm.mixin.Shadow").joinToString(","))
 | 
			
		||||
            option("NullAway:CastToNonNullMethod", "dan200.computercraft.core.util.Nullability.assertNonNull")
 | 
			
		||||
            option("NullAway:CheckOptionalEmptiness")
 | 
			
		||||
            option("NullAway:AcknowledgeRestrictiveAnnotations")
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.compileTestJava {
 | 
			
		||||
    options.errorprone {
 | 
			
		||||
        check("NullAway", CheckSeverity.OFF)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
tasks.withType(JavaCompile::class.java).configureEach {
 | 
			
		||||
    options.encoding = "UTF-8"
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.withType(AbstractArchiveTask::class.java).configureEach {
 | 
			
		||||
    isPreserveFileTimestamps = false
 | 
			
		||||
    isReproducibleFileOrder = true
 | 
			
		||||
    dirMode = Integer.valueOf("755", 8)
 | 
			
		||||
    fileMode = Integer.valueOf("664", 8)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.jar {
 | 
			
		||||
    manifest {
 | 
			
		||||
        attributes(
 | 
			
		||||
            "Specification-Title" to "computercraft",
 | 
			
		||||
            "Specification-Vendor" to "SquidDev",
 | 
			
		||||
            "Specification-Version" to "1",
 | 
			
		||||
            "Implementation-Title" to "cctweaked-${project.name}",
 | 
			
		||||
            "Implementation-Version" to modVersion,
 | 
			
		||||
            "Implementation-Vendor" to "SquidDev",
 | 
			
		||||
        )
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.javadoc {
 | 
			
		||||
    options {
 | 
			
		||||
        val stdOptions = this as StandardJavadocDocletOptions
 | 
			
		||||
        stdOptions.addBooleanOption("Xdoclint:all,-missing", true)
 | 
			
		||||
        stdOptions.links("https://docs.oracle.com/en/java/javase/17/docs/api/")
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.test {
 | 
			
		||||
    finalizedBy("jacocoTestReport")
 | 
			
		||||
 | 
			
		||||
    useJUnitPlatform()
 | 
			
		||||
    testLogging {
 | 
			
		||||
        events("skipped", "failed")
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.withType(JacocoReport::class.java).configureEach {
 | 
			
		||||
    reports.xml.required.set(true)
 | 
			
		||||
    reports.html.required.set(true)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
project.plugins.withType(CCTweakedPlugin::class.java) {
 | 
			
		||||
    // Set up jacoco to read from /all/ our source directories.
 | 
			
		||||
    val cct = project.extensions.getByType<CCTweakedExtension>()
 | 
			
		||||
    project.tasks.named("jacocoTestReport", JacocoReport::class.java) {
 | 
			
		||||
        for (ref in cct.sourceSets.get()) sourceDirectories.from(ref.allSource.sourceDirectories)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
spotless {
 | 
			
		||||
    encoding = StandardCharsets.UTF_8
 | 
			
		||||
    lineEndings = LineEnding.UNIX
 | 
			
		||||
 | 
			
		||||
    fun FormatExtension.defaults() {
 | 
			
		||||
        endWithNewline()
 | 
			
		||||
        trimTrailingWhitespace()
 | 
			
		||||
        indentWithSpaces(4)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    val licenser = LicenseHeader.create(
 | 
			
		||||
        api = rootProject.file("config/license/api.txt"),
 | 
			
		||||
        main = rootProject.file("config/license/main.txt"),
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    java {
 | 
			
		||||
        defaults()
 | 
			
		||||
        addStep(licenser)
 | 
			
		||||
        removeUnusedImports()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    val ktlintConfig = mapOf(
 | 
			
		||||
        "disabled_rules" to "no-wildcard-imports",
 | 
			
		||||
        "ij_kotlin_allow_trailing_comma" to "true",
 | 
			
		||||
        "ij_kotlin_allow_trailing_comma_on_call_site" to "true",
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    kotlinGradle {
 | 
			
		||||
        defaults()
 | 
			
		||||
        ktlint().editorConfigOverride(ktlintConfig)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    kotlin {
 | 
			
		||||
        defaults()
 | 
			
		||||
        ktlint().editorConfigOverride(ktlintConfig)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
idea.module {
 | 
			
		||||
    excludeDirs.addAll(project.files("run", "out", "logs").files)
 | 
			
		||||
 | 
			
		||||
    // Force Gradle to write to inherit the output directory from the parent, instead of writing to out/xxx/classes.
 | 
			
		||||
    // This is required for Loom, and we patch Forge's run configurations to work there.
 | 
			
		||||
    // TODO: Submit a patch to Forge to support ProjectRootManager.
 | 
			
		||||
    inheritOutputDirs = true
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,21 @@
 | 
			
		||||
import cc.tweaked.gradle.CCTweakedPlugin
 | 
			
		||||
import org.jetbrains.kotlin.gradle.tasks.KotlinCompile
 | 
			
		||||
 | 
			
		||||
plugins {
 | 
			
		||||
    kotlin("jvm")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
kotlin {
 | 
			
		||||
    jvmToolchain {
 | 
			
		||||
        languageVersion.set(CCTweakedPlugin.JAVA_VERSION)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
tasks.withType(KotlinCompile::class.java).configureEach {
 | 
			
		||||
    // So technically we shouldn't need to do this as the toolchain sets it above. However, the option only appears
 | 
			
		||||
    // to be set when the task executes, so doesn't get picked up by IDEs.
 | 
			
		||||
    kotlinOptions.jvmTarget = when {
 | 
			
		||||
        CCTweakedPlugin.JAVA_VERSION.asInt() > 8 -> CCTweakedPlugin.JAVA_VERSION.toString()
 | 
			
		||||
        else -> "1.${CCTweakedPlugin.JAVA_VERSION.asInt()}"
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										45
									
								
								buildSrc/src/main/kotlin/cc-tweaked.publishing.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								buildSrc/src/main/kotlin/cc-tweaked.publishing.gradle.kts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,45 @@
 | 
			
		||||
import org.gradle.kotlin.dsl.`maven-publish`
 | 
			
		||||
 | 
			
		||||
plugins {
 | 
			
		||||
    `java-library`
 | 
			
		||||
    `maven-publish`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
publishing {
 | 
			
		||||
    publications {
 | 
			
		||||
        register<MavenPublication>("maven") {
 | 
			
		||||
            artifactId = base.archivesName.get()
 | 
			
		||||
            from(components["java"])
 | 
			
		||||
 | 
			
		||||
            pom {
 | 
			
		||||
                name.set("CC: Tweaked")
 | 
			
		||||
                description.set("CC: Tweaked is a fork of ComputerCraft, adding programmable computers, turtles and more to Minecraft.")
 | 
			
		||||
                url.set("https://github.com/cc-tweaked/CC-Tweaked")
 | 
			
		||||
 | 
			
		||||
                scm {
 | 
			
		||||
                    url.set("https://github.com/cc-tweaked/CC-Tweaked.git")
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                issueManagement {
 | 
			
		||||
                    system.set("github")
 | 
			
		||||
                    url.set("https://github.com/cc-tweaked/CC-Tweaked/issues")
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                licenses {
 | 
			
		||||
                    license {
 | 
			
		||||
                        name.set("ComputerCraft Public License, Version 1.0")
 | 
			
		||||
                        url.set("https://github.com/cc-tweaked/CC-Tweaked/blob/HEAD/LICENSE")
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    repositories {
 | 
			
		||||
        maven("https://squiddev.cc/maven") {
 | 
			
		||||
            name = "SquidDev"
 | 
			
		||||
 | 
			
		||||
            credentials(PasswordCredentials::class)
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										31
									
								
								buildSrc/src/main/kotlin/cc-tweaked.vanilla.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								buildSrc/src/main/kotlin/cc-tweaked.vanilla.gradle.kts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,31 @@
 | 
			
		||||
/** Default configuration for non-modloader-specific Minecraft projects. */
 | 
			
		||||
 | 
			
		||||
import cc.tweaked.gradle.CCTweakedExtension
 | 
			
		||||
import cc.tweaked.gradle.CCTweakedPlugin
 | 
			
		||||
import cc.tweaked.gradle.MinecraftConfigurations
 | 
			
		||||
 | 
			
		||||
plugins {
 | 
			
		||||
    id("cc-tweaked.java-convention")
 | 
			
		||||
    id("org.spongepowered.gradle.vanilla")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
plugins.apply(CCTweakedPlugin::class.java)
 | 
			
		||||
 | 
			
		||||
val mcVersion: String by extra
 | 
			
		||||
 | 
			
		||||
minecraft {
 | 
			
		||||
    version(mcVersion)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
    val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs")
 | 
			
		||||
 | 
			
		||||
    // Depend on error prone annotations to silence a lot of compile warnings.
 | 
			
		||||
    compileOnlyApi(libs.findLibrary("errorProne.annotations").get())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
MinecraftConfigurations.setup(project)
 | 
			
		||||
 | 
			
		||||
extensions.configure(CCTweakedExtension::class.java) {
 | 
			
		||||
    linters(minecraft = true, loader = null)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										268
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/CCTweakedExtension.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										268
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/CCTweakedExtension.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,268 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import net.ltgt.gradle.errorprone.CheckSeverity
 | 
			
		||||
import net.ltgt.gradle.errorprone.errorprone
 | 
			
		||||
import org.gradle.api.GradleException
 | 
			
		||||
import org.gradle.api.NamedDomainObjectProvider
 | 
			
		||||
import org.gradle.api.Project
 | 
			
		||||
import org.gradle.api.Task
 | 
			
		||||
import org.gradle.api.attributes.TestSuiteType
 | 
			
		||||
import org.gradle.api.file.FileSystemOperations
 | 
			
		||||
import org.gradle.api.plugins.JavaPluginExtension
 | 
			
		||||
import org.gradle.api.provider.Provider
 | 
			
		||||
import org.gradle.api.provider.SetProperty
 | 
			
		||||
import org.gradle.api.reporting.ReportingExtension
 | 
			
		||||
import org.gradle.api.tasks.SourceSet
 | 
			
		||||
import org.gradle.api.tasks.bundling.Jar
 | 
			
		||||
import org.gradle.api.tasks.compile.JavaCompile
 | 
			
		||||
import org.gradle.api.tasks.javadoc.Javadoc
 | 
			
		||||
import org.gradle.configurationcache.extensions.capitalized
 | 
			
		||||
import org.gradle.language.base.plugins.LifecycleBasePlugin
 | 
			
		||||
import org.gradle.language.jvm.tasks.ProcessResources
 | 
			
		||||
import org.gradle.process.JavaForkOptions
 | 
			
		||||
import org.gradle.testing.jacoco.plugins.JacocoCoverageReport
 | 
			
		||||
import org.gradle.testing.jacoco.plugins.JacocoPluginExtension
 | 
			
		||||
import org.gradle.testing.jacoco.plugins.JacocoTaskExtension
 | 
			
		||||
import org.gradle.testing.jacoco.tasks.JacocoReport
 | 
			
		||||
import org.jetbrains.kotlin.gradle.dsl.KotlinProjectExtension
 | 
			
		||||
import org.jetbrains.kotlin.gradle.tasks.KotlinCompile
 | 
			
		||||
import java.io.File
 | 
			
		||||
import java.io.IOException
 | 
			
		||||
import java.net.URI
 | 
			
		||||
import java.net.URL
 | 
			
		||||
import java.util.regex.Pattern
 | 
			
		||||
 | 
			
		||||
abstract class CCTweakedExtension(
 | 
			
		||||
    private val project: Project,
 | 
			
		||||
    private val fs: FileSystemOperations,
 | 
			
		||||
) {
 | 
			
		||||
    /** Get the hash of the latest git commit. */
 | 
			
		||||
    val gitHash: Provider<String> = gitProvider(project, "<no git hash>") {
 | 
			
		||||
        ProcessHelpers.captureOut("git", "-C", project.rootProject.projectDir.absolutePath, "rev-parse", "HEAD").trim()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /** Get the current git branch. */
 | 
			
		||||
    val gitBranch: Provider<String> = gitProvider(project, "<no git branch>") {
 | 
			
		||||
        ProcessHelpers.captureOut("git", "-C", project.rootProject.projectDir.absolutePath, "rev-parse", "--abbrev-ref", "HEAD")
 | 
			
		||||
            .trim()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /** Get a list of all contributors to the project. */
 | 
			
		||||
    val gitContributors: Provider<List<String>> = gitProvider(project, listOf()) {
 | 
			
		||||
        ProcessHelpers.captureLines(
 | 
			
		||||
            "git", "-C", project.rootProject.projectDir.absolutePath, "shortlog", "-ns",
 | 
			
		||||
            "--group=author", "--group=trailer:co-authored-by", "HEAD",
 | 
			
		||||
        )
 | 
			
		||||
            .asSequence()
 | 
			
		||||
            .map {
 | 
			
		||||
                val matcher = COMMIT_COUNTS.matcher(it)
 | 
			
		||||
                matcher.find()
 | 
			
		||||
                matcher.group(1)
 | 
			
		||||
            }
 | 
			
		||||
            .filter { !IGNORED_USERS.contains(it) }
 | 
			
		||||
            .toList()
 | 
			
		||||
            .sortedWith(String.CASE_INSENSITIVE_ORDER)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * References to other sources
 | 
			
		||||
     */
 | 
			
		||||
    val sourceDirectories: SetProperty<SourceSetReference> = project.objects.setProperty(SourceSetReference::class.java)
 | 
			
		||||
 | 
			
		||||
    /** All source sets referenced by this project. */
 | 
			
		||||
    val sourceSets = sourceDirectories.map { x -> x.map { it.sourceSet } }
 | 
			
		||||
 | 
			
		||||
    init {
 | 
			
		||||
        sourceDirectories.finalizeValueOnRead()
 | 
			
		||||
        project.afterEvaluate { sourceDirectories.disallowChanges() }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Mark this project as consuming another project. Its [sourceDirectories] are added, allowing easier configuration
 | 
			
		||||
     * of run configurations and other tasks which consume sources/classes.
 | 
			
		||||
     */
 | 
			
		||||
    fun externalSources(project: Project) {
 | 
			
		||||
        val otherCct = project.extensions.getByType(CCTweakedExtension::class.java)
 | 
			
		||||
        for (sourceSet in otherCct.sourceDirectories.get()) {
 | 
			
		||||
            sourceDirectories.add(SourceSetReference(sourceSet.sourceSet, classes = sourceSet.classes, external = true))
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Add a dependency on another project such that its sources and compiles are processed with this one.
 | 
			
		||||
     *
 | 
			
		||||
     * This is used when importing a common library into a loader-specific one, as we want to compile sources using
 | 
			
		||||
     * the loader-specific sources.
 | 
			
		||||
     */
 | 
			
		||||
    fun inlineProject(path: String) {
 | 
			
		||||
        val otherProject = project.evaluationDependsOn(path)
 | 
			
		||||
        val otherJava = otherProject.extensions.getByType(JavaPluginExtension::class.java)
 | 
			
		||||
        val main = otherJava.sourceSets.getByName("main")
 | 
			
		||||
        val client = otherJava.sourceSets.getByName("client")
 | 
			
		||||
        val testMod = otherJava.sourceSets.findByName("testMod")
 | 
			
		||||
        val testFixtures = otherJava.sourceSets.findByName("testFixtures")
 | 
			
		||||
 | 
			
		||||
        // Pull in sources from the other project.
 | 
			
		||||
        extendSourceSet(otherProject, main)
 | 
			
		||||
        extendSourceSet(otherProject, client)
 | 
			
		||||
        if (testMod != null) extendSourceSet(otherProject, testMod)
 | 
			
		||||
        if (testFixtures != null) extendSourceSet(otherProject, testFixtures)
 | 
			
		||||
 | 
			
		||||
        // The extra source-processing tasks should include these files too.
 | 
			
		||||
        project.tasks.named(main.javadocTaskName, Javadoc::class.java) { source(main.allJava, client.allJava) }
 | 
			
		||||
        project.tasks.named(main.sourcesJarTaskName, Jar::class.java) { from(main.allSource, client.allSource) }
 | 
			
		||||
        sourceDirectories.addAll(SourceSetReference.inline(main), SourceSetReference.inline(client))
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Extend a source set with files from another project.
 | 
			
		||||
     *
 | 
			
		||||
     * This actually extends the original compile tasks, as extending the source sets does not play well with IDEs.
 | 
			
		||||
     */
 | 
			
		||||
    private fun extendSourceSet(otherProject: Project, sourceSet: SourceSet) {
 | 
			
		||||
        project.tasks.named(sourceSet.compileJavaTaskName, JavaCompile::class.java) {
 | 
			
		||||
            dependsOn(otherProject.tasks.named(sourceSet.compileJavaTaskName)) // Avoid duplicate compile errors
 | 
			
		||||
            source(sourceSet.allJava)
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        project.tasks.named(sourceSet.processResourcesTaskName, ProcessResources::class.java) {
 | 
			
		||||
            from(sourceSet.resources)
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // Also try to depend on Kotlin if it exists
 | 
			
		||||
        val kotlin = otherProject.extensions.findByType(KotlinProjectExtension::class.java)
 | 
			
		||||
        if (kotlin != null) {
 | 
			
		||||
            val compileKotlin = sourceSet.getCompileTaskName("kotlin")
 | 
			
		||||
            project.tasks.named(compileKotlin, KotlinCompile::class.java) {
 | 
			
		||||
                dependsOn(otherProject.tasks.named(compileKotlin))
 | 
			
		||||
                source(kotlin.sourceSets.getByName(sourceSet.name).kotlin)
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // If we're doing an IDE sync, add a fake dependency to ensure it's on the classpath.
 | 
			
		||||
        if (isIdeSync) project.dependencies.add(sourceSet.apiConfigurationName, sourceSet.output)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun linters(@Suppress("UNUSED_PARAMETER") vararg unused: UseNamedArgs, minecraft: Boolean, loader: String?) {
 | 
			
		||||
        val java = project.extensions.getByType(JavaPluginExtension::class.java)
 | 
			
		||||
        val sourceSets = java.sourceSets
 | 
			
		||||
 | 
			
		||||
        project.dependencies.run { add("errorprone", project(mapOf("path" to ":lints"))) }
 | 
			
		||||
        sourceSets.all {
 | 
			
		||||
            val name = name
 | 
			
		||||
            project.tasks.named(compileJavaTaskName, JavaCompile::class.java) {
 | 
			
		||||
                options.errorprone {
 | 
			
		||||
                    // Only the main source set should run the side checker
 | 
			
		||||
                    check("SideChecker", if (minecraft && name == "main") CheckSeverity.DEFAULT else CheckSeverity.OFF)
 | 
			
		||||
 | 
			
		||||
                    // The MissingLoaderOverride check superseeds the MissingOverride one, so disable that.
 | 
			
		||||
                    if (loader != null) {
 | 
			
		||||
                        check("MissingOverride", CheckSeverity.OFF)
 | 
			
		||||
                        option("ModLoader", loader)
 | 
			
		||||
                    } else {
 | 
			
		||||
                        check("LoaderOverride", CheckSeverity.OFF)
 | 
			
		||||
                        check("MissingLoaderOverride", CheckSeverity.OFF)
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun <T> jacoco(task: NamedDomainObjectProvider<T>) where T : Task, T : JavaForkOptions {
 | 
			
		||||
        val classDump = project.buildDir.resolve("jacocoClassDump/${task.name}")
 | 
			
		||||
        val reportTaskName = "jacoco${task.name.capitalized()}Report"
 | 
			
		||||
 | 
			
		||||
        val jacoco = project.extensions.getByType(JacocoPluginExtension::class.java)
 | 
			
		||||
        task.configure {
 | 
			
		||||
            finalizedBy(reportTaskName)
 | 
			
		||||
 | 
			
		||||
            doFirst("Clean class dump directory") { fs.delete { delete(classDump) } }
 | 
			
		||||
 | 
			
		||||
            jacoco.applyTo(this)
 | 
			
		||||
            extensions.configure(JacocoTaskExtension::class.java) {
 | 
			
		||||
                includes = listOf("dan200.computercraft.*")
 | 
			
		||||
                classDumpDir = classDump
 | 
			
		||||
 | 
			
		||||
                // Older versions of modlauncher don't include a protection domain (and thus no code
 | 
			
		||||
                // source). Jacoco skips such classes by default, so we need to explicitly include them.
 | 
			
		||||
                isIncludeNoLocationClasses = true
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        project.tasks.register(reportTaskName, JacocoReport::class.java) {
 | 
			
		||||
            group = LifecycleBasePlugin.VERIFICATION_GROUP
 | 
			
		||||
            description = "Generates code coverage report for the ${task.name} task."
 | 
			
		||||
 | 
			
		||||
            executionData(task.get())
 | 
			
		||||
            classDirectories.from(classDump)
 | 
			
		||||
 | 
			
		||||
            // Don't want to use sourceSets(...) here as we have a custom class directory.
 | 
			
		||||
            for (ref in sourceSets.get()) sourceDirectories.from(ref.allSource.sourceDirectories)
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        project.extensions.configure(ReportingExtension::class.java) {
 | 
			
		||||
            reports.register("${task.name}CodeCoverageReport", JacocoCoverageReport::class.java) {
 | 
			
		||||
                testType.set(TestSuiteType.INTEGRATION_TEST)
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Download a file by creating a dummy Ivy repository.
 | 
			
		||||
     *
 | 
			
		||||
     * This should only be used for one-off downloads. Using a more conventional Ivy or Maven repository is preferred
 | 
			
		||||
     * where possible.
 | 
			
		||||
     */
 | 
			
		||||
    fun downloadFile(label: String, url: String): File {
 | 
			
		||||
        val url = URL(url)
 | 
			
		||||
        val path = File(url.path)
 | 
			
		||||
 | 
			
		||||
        project.repositories.ivy {
 | 
			
		||||
            name = label
 | 
			
		||||
            setUrl(URI(url.protocol, url.userInfo, url.host, url.port, path.parent, null, null))
 | 
			
		||||
            patternLayout {
 | 
			
		||||
                artifact("[artifact].[ext]")
 | 
			
		||||
            }
 | 
			
		||||
            metadataSources {
 | 
			
		||||
                artifact()
 | 
			
		||||
            }
 | 
			
		||||
            content {
 | 
			
		||||
                includeModule("cc.tweaked.internal", path.nameWithoutExtension)
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return project.configurations.detachedConfiguration(
 | 
			
		||||
            project.dependencies.create(
 | 
			
		||||
                mapOf(
 | 
			
		||||
                    "group" to "cc.tweaked.internal",
 | 
			
		||||
                    "name" to path.nameWithoutExtension,
 | 
			
		||||
                    "ext" to path.extension,
 | 
			
		||||
                ),
 | 
			
		||||
            ),
 | 
			
		||||
        ).resolve().single()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    companion object {
 | 
			
		||||
        private val COMMIT_COUNTS = Pattern.compile("""^\s*[0-9]+\s+(.*)$""")
 | 
			
		||||
        private val IGNORED_USERS = setOf(
 | 
			
		||||
            "GitHub", "Daniel Ratcliffe", "Weblate",
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
        private fun <T> gitProvider(project: Project, default: T, supplier: () -> T): Provider<T> {
 | 
			
		||||
            return project.provider {
 | 
			
		||||
                try {
 | 
			
		||||
                    supplier()
 | 
			
		||||
                } catch (e: IOException) {
 | 
			
		||||
                    project.logger.error("Cannot read Git repository: ${e.message}")
 | 
			
		||||
                    default
 | 
			
		||||
                } catch (e: GradleException) {
 | 
			
		||||
                    project.logger.error("Cannot read Git repository: ${e.message}")
 | 
			
		||||
                    default
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        private val isIdeSync: Boolean
 | 
			
		||||
            get() = java.lang.Boolean.parseBoolean(System.getProperty("idea.sync.active", "false"))
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,25 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.Plugin
 | 
			
		||||
import org.gradle.api.Project
 | 
			
		||||
import org.gradle.api.plugins.JavaPlugin
 | 
			
		||||
import org.gradle.api.plugins.JavaPluginExtension
 | 
			
		||||
import org.gradle.jvm.toolchain.JavaLanguageVersion
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Configures projects to match a shared configuration.
 | 
			
		||||
 */
 | 
			
		||||
class CCTweakedPlugin : Plugin<Project> {
 | 
			
		||||
    override fun apply(project: Project) {
 | 
			
		||||
        val cct = project.extensions.create("cct", CCTweakedExtension::class.java)
 | 
			
		||||
 | 
			
		||||
        project.plugins.withType(JavaPlugin::class.java) {
 | 
			
		||||
            val sourceSets = project.extensions.getByType(JavaPluginExtension::class.java).sourceSets
 | 
			
		||||
            cct.sourceDirectories.add(SourceSetReference.internal(sourceSets.getByName("main")))
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    companion object {
 | 
			
		||||
        val JAVA_VERSION = JavaLanguageVersion.of(17)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										64
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/CheckChangelog.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/CheckChangelog.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,64 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.DefaultTask
 | 
			
		||||
import org.gradle.api.GradleException
 | 
			
		||||
import org.gradle.api.file.RegularFileProperty
 | 
			
		||||
import org.gradle.api.provider.Property
 | 
			
		||||
import org.gradle.api.tasks.*
 | 
			
		||||
import org.gradle.language.base.plugins.LifecycleBasePlugin
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Checks the `changelog.md` and `whatsnew.md` files are well-formed.
 | 
			
		||||
 */
 | 
			
		||||
@CacheableTask
 | 
			
		||||
abstract class CheckChangelog : DefaultTask() {
 | 
			
		||||
    init {
 | 
			
		||||
        group = LifecycleBasePlugin.VERIFICATION_GROUP
 | 
			
		||||
        description = "Verifies the changelog and whatsnew file are consistent."
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @get:Input
 | 
			
		||||
    abstract val version: Property<String>
 | 
			
		||||
 | 
			
		||||
    @get:InputFile
 | 
			
		||||
    @get:PathSensitive(PathSensitivity.NONE)
 | 
			
		||||
    abstract val changelog: RegularFileProperty
 | 
			
		||||
 | 
			
		||||
    @get:InputFile
 | 
			
		||||
    @get:PathSensitive(PathSensitivity.NONE)
 | 
			
		||||
    abstract val whatsNew: RegularFileProperty
 | 
			
		||||
 | 
			
		||||
    @TaskAction
 | 
			
		||||
    fun check() {
 | 
			
		||||
        val version = version.get()
 | 
			
		||||
 | 
			
		||||
        var ok = true
 | 
			
		||||
 | 
			
		||||
        // Check we're targetting the current version
 | 
			
		||||
        var whatsNew = whatsNew.get().asFile.readLines()
 | 
			
		||||
        if (whatsNew[0] != "New features in CC: Tweaked $version") {
 | 
			
		||||
            ok = false
 | 
			
		||||
            logger.error("Expected `whatsnew.md' to target $version.")
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // Check "read more" exists and trim it
 | 
			
		||||
        val idx = whatsNew.indexOfFirst { it == "Type \"help changelog\" to see the full version history." }
 | 
			
		||||
        if (idx == -1) {
 | 
			
		||||
            ok = false
 | 
			
		||||
            logger.error("Must mention the changelog in whatsnew.md")
 | 
			
		||||
        } else {
 | 
			
		||||
            whatsNew = whatsNew.slice(0 until idx)
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // Check whatsnew and changelog match.
 | 
			
		||||
        val expectedChangelog = sequenceOf("# ${whatsNew[0]}") + whatsNew.slice(1 until whatsNew.size).asSequence()
 | 
			
		||||
        val changelog = changelog.get().asFile.readLines()
 | 
			
		||||
        val mismatch = expectedChangelog.zip(changelog.asSequence()).filter { (a, b) -> a != b }.firstOrNull()
 | 
			
		||||
        if (mismatch != null) {
 | 
			
		||||
            ok = false
 | 
			
		||||
            logger.error("whatsnew and changelog are not in sync")
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (!ok) throw GradleException("Could not check release")
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										72
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/CheckLicense.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/CheckLicense.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,72 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import com.diffplug.spotless.FormatterFunc
 | 
			
		||||
import com.diffplug.spotless.FormatterStep
 | 
			
		||||
import com.diffplug.spotless.generic.LicenseHeaderStep
 | 
			
		||||
import java.io.File
 | 
			
		||||
import java.io.Serializable
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Similar to [LicenseHeaderStep], but supports multiple licenses.
 | 
			
		||||
 */
 | 
			
		||||
object LicenseHeader {
 | 
			
		||||
    /**
 | 
			
		||||
     * The current year to use in templates. Intentionally not dynamic to avoid failing the build.
 | 
			
		||||
     */
 | 
			
		||||
    private const val YEAR = 2022
 | 
			
		||||
 | 
			
		||||
    private val COMMENT = Regex("""^/\*(.*?)\*/\n?""", RegexOption.DOT_MATCHES_ALL)
 | 
			
		||||
 | 
			
		||||
    fun create(api: File, main: File): FormatterStep = FormatterStep.createLazy(
 | 
			
		||||
        "license",
 | 
			
		||||
        { Licenses(getTemplateText(api), getTemplateText(main)) },
 | 
			
		||||
        { state -> FormatterFunc.NeedsFile { contents, file -> formatFile(state, contents, file) } },
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    private fun getTemplateText(file: File): String =
 | 
			
		||||
        file.readText().trim().replace("\${year}", "$YEAR")
 | 
			
		||||
 | 
			
		||||
    private fun formatFile(licenses: Licenses, contents: String, file: File): String {
 | 
			
		||||
        val license = getLicense(contents)
 | 
			
		||||
        val expectedLicense = getExpectedLicense(licenses, file.parentFile)
 | 
			
		||||
 | 
			
		||||
        return when {
 | 
			
		||||
            license == null -> setLicense(expectedLicense, contents)
 | 
			
		||||
            license.second != expectedLicense -> setLicense(expectedLicense, contents, license.first)
 | 
			
		||||
            else -> contents
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun getExpectedLicense(licenses: Licenses, root: File): String {
 | 
			
		||||
        var file: File? = root
 | 
			
		||||
        while (file != null) {
 | 
			
		||||
            if (file.name == "api" && file.parentFile?.name == "computercraft") return licenses.api
 | 
			
		||||
            file = file.parentFile
 | 
			
		||||
        }
 | 
			
		||||
        return licenses.main
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun getLicense(contents: String): Pair<Int, String>? {
 | 
			
		||||
        val match = COMMENT.find(contents) ?: return null
 | 
			
		||||
        val license = match.groups[1]!!.value
 | 
			
		||||
            .trim().lineSequence()
 | 
			
		||||
            .map { it.trimStart(' ', '*') }
 | 
			
		||||
            .joinToString("\n")
 | 
			
		||||
        return Pair(match.range.last + 1, license)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun setLicense(license: String, contents: String, start: Int = 0): String {
 | 
			
		||||
        val out = StringBuilder()
 | 
			
		||||
        out.append("/*\n")
 | 
			
		||||
        for (line in license.lineSequence()) out.append(" * ").append(line).append("\n")
 | 
			
		||||
        out.append(" */\n")
 | 
			
		||||
        out.append(contents, start, contents.length)
 | 
			
		||||
        return out.toString()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private data class Licenses(val api: String, val main: String) : Serializable {
 | 
			
		||||
        companion object {
 | 
			
		||||
            private const val serialVersionUID: Long = 7741106448372435662L
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										11
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/ExecTasks.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/ExecTasks.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,11 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.provider.Property
 | 
			
		||||
import org.gradle.api.tasks.AbstractExecTask
 | 
			
		||||
import org.gradle.api.tasks.OutputDirectory
 | 
			
		||||
import java.io.File
 | 
			
		||||
 | 
			
		||||
abstract class ExecToDir : AbstractExecTask<ExecToDir>(ExecToDir::class.java) {
 | 
			
		||||
    @get:OutputDirectory
 | 
			
		||||
    abstract val output: Property<File>
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										117
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/Extensions.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/Extensions.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,117 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.artifacts.dsl.DependencyHandler
 | 
			
		||||
import org.gradle.api.tasks.JavaExec
 | 
			
		||||
import org.gradle.process.BaseExecSpec
 | 
			
		||||
import org.gradle.process.JavaExecSpec
 | 
			
		||||
import org.gradle.process.ProcessForkOptions
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Add an annotation processor to all source sets.
 | 
			
		||||
 */
 | 
			
		||||
fun DependencyHandler.annotationProcessorEverywhere(dep: Any) {
 | 
			
		||||
    add("compileOnly", dep)
 | 
			
		||||
    add("annotationProcessor", dep)
 | 
			
		||||
 | 
			
		||||
    add("clientCompileOnly", dep)
 | 
			
		||||
    add("clientAnnotationProcessor", dep)
 | 
			
		||||
 | 
			
		||||
    add("testCompileOnly", dep)
 | 
			
		||||
    add("testAnnotationProcessor", dep)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * A version of [JavaExecSpec.copyTo] which copies *all* properties.
 | 
			
		||||
 */
 | 
			
		||||
fun JavaExec.copyToFull(spec: JavaExec) {
 | 
			
		||||
    copyTo(spec)
 | 
			
		||||
 | 
			
		||||
    // Additional Java options
 | 
			
		||||
    spec.jvmArgs = jvmArgs // Fabric overrides getJvmArgs so copyTo doesn't do the right thing.
 | 
			
		||||
    spec.args = args
 | 
			
		||||
    spec.argumentProviders.addAll(argumentProviders)
 | 
			
		||||
    spec.mainClass.set(mainClass)
 | 
			
		||||
    spec.classpath = classpath
 | 
			
		||||
    spec.javaLauncher.set(javaLauncher)
 | 
			
		||||
    if (executable != null) spec.setExecutable(executable!!)
 | 
			
		||||
 | 
			
		||||
    // Additional ExecSpec options
 | 
			
		||||
    copyToExec(spec)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Copy additional [BaseExecSpec] options which aren't handled by [ProcessForkOptions.copyTo].
 | 
			
		||||
 */
 | 
			
		||||
fun BaseExecSpec.copyToExec(spec: BaseExecSpec) {
 | 
			
		||||
    spec.isIgnoreExitValue = isIgnoreExitValue
 | 
			
		||||
    if (standardInput != null) spec.standardInput = standardInput
 | 
			
		||||
    if (standardOutput != null) spec.standardOutput = standardOutput
 | 
			
		||||
    if (errorOutput != null) spec.errorOutput = errorOutput
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * An alternative to [Nothing] with a more descriptive name. Use to enforce calling a function with named arguments:
 | 
			
		||||
 *
 | 
			
		||||
 * ```kotlin
 | 
			
		||||
 * fun f(vararg unused: UseNamedArgs, arg1: Int, arg2: Int) {
 | 
			
		||||
 *   // ...
 | 
			
		||||
 * }
 | 
			
		||||
 * ```
 | 
			
		||||
 */
 | 
			
		||||
class UseNamedArgs private constructor()
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * An [AutoCloseable] implementation which can be used to combine other [AutoCloseable] instances.
 | 
			
		||||
 *
 | 
			
		||||
 * Values which implement [AutoCloseable] can be dynamically registered with [CloseScope.add]. When the scope is closed,
 | 
			
		||||
 * each value is closed in the opposite order.
 | 
			
		||||
 *
 | 
			
		||||
 * This is largely intended for cases where it's not appropriate to nest [AutoCloseable.use], for instance when nested
 | 
			
		||||
 * would be too deep.
 | 
			
		||||
 */
 | 
			
		||||
class CloseScope : AutoCloseable {
 | 
			
		||||
    private val toClose = ArrayDeque<AutoCloseable>()
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Add a value to be closed when this scope is closed.
 | 
			
		||||
     */
 | 
			
		||||
    public fun add(value: AutoCloseable) {
 | 
			
		||||
        toClose.addLast(value)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun close() {
 | 
			
		||||
        close(null)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @PublishedApi
 | 
			
		||||
    internal fun close(baseException: Throwable?) {
 | 
			
		||||
        var exception = baseException
 | 
			
		||||
 | 
			
		||||
        while (true) {
 | 
			
		||||
            var toClose = toClose.removeLastOrNull() ?: break
 | 
			
		||||
            try {
 | 
			
		||||
                toClose.close()
 | 
			
		||||
            } catch (e: Throwable) {
 | 
			
		||||
                if (exception == null) {
 | 
			
		||||
                    exception = e
 | 
			
		||||
                } else {
 | 
			
		||||
                    exception.addSuppressed(e)
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (exception != null) throw exception
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    inline fun <R> use(block: (CloseScope) -> R): R {
 | 
			
		||||
        var exception: Throwable? = null
 | 
			
		||||
        try {
 | 
			
		||||
            return block(this)
 | 
			
		||||
        } catch (e: Throwable) {
 | 
			
		||||
            exception = e
 | 
			
		||||
            throw e
 | 
			
		||||
        } finally {
 | 
			
		||||
            close(exception)
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,170 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.Project
 | 
			
		||||
import org.gradle.api.logging.Logging
 | 
			
		||||
import org.w3c.dom.Attr
 | 
			
		||||
import org.w3c.dom.Document
 | 
			
		||||
import org.w3c.dom.Node
 | 
			
		||||
import org.xml.sax.InputSource
 | 
			
		||||
import java.nio.file.Files
 | 
			
		||||
import java.nio.file.Path
 | 
			
		||||
import javax.xml.parsers.DocumentBuilderFactory
 | 
			
		||||
import javax.xml.transform.TransformerFactory
 | 
			
		||||
import javax.xml.transform.dom.DOMSource
 | 
			
		||||
import javax.xml.transform.stream.StreamResult
 | 
			
		||||
import javax.xml.xpath.XPathConstants
 | 
			
		||||
import javax.xml.xpath.XPathFactory
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Patches up run configurations from ForgeGradle and Loom.
 | 
			
		||||
 *
 | 
			
		||||
 * Would be good to PR some (or all) of these changes upstream at some point.
 | 
			
		||||
 *
 | 
			
		||||
 * @see net.fabricmc.loom.configuration.ide.idea.IdeaSyncTask
 | 
			
		||||
 * @see net.minecraftforge.gradle.common.util.runs.IntellijRunGenerator
 | 
			
		||||
 */
 | 
			
		||||
internal class IdeaRunConfigurations(project: Project) {
 | 
			
		||||
    private val rootProject = project.rootProject
 | 
			
		||||
 | 
			
		||||
    private val documentBuilder = DocumentBuilderFactory.newInstance().newDocumentBuilder()
 | 
			
		||||
    private val xpath = XPathFactory.newInstance().newXPath()
 | 
			
		||||
    private val writer = TransformerFactory.newInstance().newTransformer()
 | 
			
		||||
 | 
			
		||||
    private val ideaDir = rootProject.file(".idea/")
 | 
			
		||||
    private val buildDir: Lazy<String?> = lazy {
 | 
			
		||||
        val ideaMisc = ideaDir.resolve("misc.xml")
 | 
			
		||||
 | 
			
		||||
        try {
 | 
			
		||||
            val doc = Files.newBufferedReader(ideaMisc.toPath()).use {
 | 
			
		||||
                documentBuilder.parse(InputSource(it))
 | 
			
		||||
            }
 | 
			
		||||
            val node =
 | 
			
		||||
                xpath.evaluate("//component[@name=\"ProjectRootManager\"]/output", doc, XPathConstants.NODE) as Node
 | 
			
		||||
            val attr = node.attributes.getNamedItem("url") as Attr
 | 
			
		||||
            attr.value.removePrefix("file://")
 | 
			
		||||
        } catch (e: Exception) {
 | 
			
		||||
            LOGGER.error("Failed to find root directory", e)
 | 
			
		||||
            null
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun patch() = synchronized(LOCK) {
 | 
			
		||||
        val runConfigDir = ideaDir.resolve("runConfigurations")
 | 
			
		||||
        if (!runConfigDir.isDirectory) return
 | 
			
		||||
 | 
			
		||||
        Files.list(runConfigDir.toPath()).use {
 | 
			
		||||
            for (configuration in it) {
 | 
			
		||||
                val filename = configuration.fileName.toString();
 | 
			
		||||
                when {
 | 
			
		||||
                    filename.endsWith("_fabric.xml") -> patchFabric(configuration)
 | 
			
		||||
                    filename.startsWith("forge_") && filename.endsWith(".xml") -> patchForge(configuration)
 | 
			
		||||
                    else -> {}
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun patchFabric(path: Path) = withXml(path) {
 | 
			
		||||
        setXml("//configuration", "folderName") { "Fabric" }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun patchForge(path: Path) = withXml(path) {
 | 
			
		||||
        val configId = path.fileName.toString().removePrefix("forge_").removeSuffix(".xml")
 | 
			
		||||
        val sourceSet = forgeConfigs[configId]
 | 
			
		||||
        if (sourceSet == null) {
 | 
			
		||||
            LOGGER.error("[{}] Cannot map run configuration to a known source set", path)
 | 
			
		||||
            return@withXml
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        setXml("//configuration", "folderName") { "Forge" }
 | 
			
		||||
        setXml("//configuration/module", "name") { "${rootProject.name}.forge.$sourceSet" }
 | 
			
		||||
 | 
			
		||||
        if (buildDir.value == null) return@withXml
 | 
			
		||||
        setXml("//configuration/envs/env[@name=\"MOD_CLASSES\"]", "value") { classpath ->
 | 
			
		||||
            val classes = classpath!!.split(':')
 | 
			
		||||
            val newClasses = mutableListOf<String>()
 | 
			
		||||
            fun appendUnique(x: String) {
 | 
			
		||||
                if (!newClasses.contains(x)) newClasses.add(x)
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            for (entry in classes) {
 | 
			
		||||
                if (!entry.contains("/out/")) {
 | 
			
		||||
                    appendUnique(entry)
 | 
			
		||||
                    continue
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                val match = CLASSPATH_ENTRY.matchEntire(entry)
 | 
			
		||||
                if (match != null) {
 | 
			
		||||
                    val modId = match.groups["modId"]!!.value
 | 
			
		||||
                    val proj = match.groups["proj"]!!.value
 | 
			
		||||
                    var component = match.groups["component"]!!.value
 | 
			
		||||
                    if (component == "production") component = "main"
 | 
			
		||||
 | 
			
		||||
                    appendUnique(forgeModEntry(modId, proj, component))
 | 
			
		||||
                } else {
 | 
			
		||||
                    LOGGER.warn("[{}] Unknown classpath entry {}", path, entry)
 | 
			
		||||
                    appendUnique(entry)
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            // Ensure common code is on the classpath
 | 
			
		||||
            for (proj in listOf("common", "common-api")) {
 | 
			
		||||
                for (component in listOf("main", "client")) {
 | 
			
		||||
                    appendUnique(forgeModEntry("computercraft", proj, component))
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (newClasses.any { it.startsWith("cctest%%") }) {
 | 
			
		||||
                appendUnique(forgeModEntry("cctest", "core", "testFixtures"))
 | 
			
		||||
                appendUnique(forgeModEntry("cctest", "common", "testFixtures"))
 | 
			
		||||
                appendUnique(forgeModEntry("cctest", "common", "testMod"))
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            newClasses.joinToString(":")
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun forgeModEntry(mod: String, project: String, component: String) =
 | 
			
		||||
        "$mod%%${buildDir.value}/production/${rootProject.name}.$project.$component"
 | 
			
		||||
 | 
			
		||||
    private fun LocatedDocument.setXml(xpath: String, attribute: String, value: (String?) -> String) {
 | 
			
		||||
        val node = this@IdeaRunConfigurations.xpath.evaluate(xpath, document, XPathConstants.NODE) as Node?
 | 
			
		||||
        if (node == null) {
 | 
			
		||||
            LOGGER.error("[{}] Cannot find {}", path.fileName, xpath)
 | 
			
		||||
            return
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        val attr = node.attributes.getNamedItem(attribute) as Attr? ?: document.createAttribute(attribute)
 | 
			
		||||
        val oldValue = attr.value
 | 
			
		||||
        attr.value = value(attr.value)
 | 
			
		||||
        node.attributes.setNamedItem(attr)
 | 
			
		||||
 | 
			
		||||
        if (oldValue != attr.value) {
 | 
			
		||||
            LOGGER.info("[{}] Setting {}@{}:\n  Old: {}\n  New: {}", path.fileName, xpath, attribute, oldValue, attr.value)
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun withXml(path: Path, run: LocatedDocument.() -> Unit) {
 | 
			
		||||
        val doc = Files.newBufferedReader(path).use { documentBuilder.parse(InputSource(it)) }
 | 
			
		||||
        run(LocatedDocument(path, doc))
 | 
			
		||||
        Files.newBufferedWriter(path).use { writer.transform(DOMSource(doc), StreamResult(it)) }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private class LocatedDocument(val path: Path, val document: Document)
 | 
			
		||||
 | 
			
		||||
    companion object {
 | 
			
		||||
        private val LOGGER = Logging.getLogger(IdeaRunConfigurations::class.java)
 | 
			
		||||
        private val LOCK = Any()
 | 
			
		||||
 | 
			
		||||
        private val CLASSPATH_ENTRY =
 | 
			
		||||
            Regex("(?<modId>[a-z]+)%%\\\$PROJECT_DIR\\\$/projects/(?<proj>[a-z-]+)/out/(?<component>\\w+)/(?<type>[a-z]+)\$")
 | 
			
		||||
 | 
			
		||||
        private val forgeConfigs = mapOf(
 | 
			
		||||
            "runClient" to "client",
 | 
			
		||||
            "runData" to "main",
 | 
			
		||||
            "runGameTestServer" to "testMod",
 | 
			
		||||
            "runServer" to "main",
 | 
			
		||||
            "runTestClient" to "testMod",
 | 
			
		||||
        )
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										123
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/Illuaminate.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/Illuaminate.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,123 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.DefaultTask
 | 
			
		||||
import org.gradle.api.Plugin
 | 
			
		||||
import org.gradle.api.Project
 | 
			
		||||
import org.gradle.api.Task
 | 
			
		||||
import org.gradle.api.artifacts.Dependency
 | 
			
		||||
import org.gradle.api.provider.Property
 | 
			
		||||
import org.gradle.api.provider.Provider
 | 
			
		||||
import org.gradle.api.tasks.AbstractExecTask
 | 
			
		||||
import org.gradle.api.tasks.Input
 | 
			
		||||
import org.gradle.api.tasks.TaskAction
 | 
			
		||||
import java.io.File
 | 
			
		||||
 | 
			
		||||
abstract class IlluaminateExtension {
 | 
			
		||||
    /** The version of illuaminate to use. */
 | 
			
		||||
    abstract val version: Property<String>
 | 
			
		||||
 | 
			
		||||
    /** The path to illuaminate. If not given, illuaminate will be downloaded automatically. */
 | 
			
		||||
    abstract val file: Property<File>
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
class IlluaminatePlugin : Plugin<Project> {
 | 
			
		||||
    override fun apply(project: Project) {
 | 
			
		||||
        val extension = project.extensions.create("illuaminate", IlluaminateExtension::class.java)
 | 
			
		||||
        extension.file.convention(setupDependency(project, extension.version))
 | 
			
		||||
 | 
			
		||||
        project.tasks.register(SetupIlluaminate.NAME, SetupIlluaminate::class.java) {
 | 
			
		||||
            file.set(extension.file.map { it.absolutePath })
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /** Set up a repository for illuaminate and download our binary from it. */
 | 
			
		||||
    private fun setupDependency(project: Project, version: Provider<String>): Provider<File> {
 | 
			
		||||
        project.repositories.ivy {
 | 
			
		||||
            name = "Illuaminate"
 | 
			
		||||
            setUrl("https://squiddev.cc/illuaminate/bin/")
 | 
			
		||||
            patternLayout {
 | 
			
		||||
                artifact("[revision]/[artifact]-[ext]")
 | 
			
		||||
            }
 | 
			
		||||
            metadataSources {
 | 
			
		||||
                artifact()
 | 
			
		||||
            }
 | 
			
		||||
            content {
 | 
			
		||||
                includeModule("cc.squiddev", "illuaminate")
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return version.map {
 | 
			
		||||
            val dep = illuaminateArtifact(project, it)
 | 
			
		||||
            val configuration = project.configurations.detachedConfiguration(dep)
 | 
			
		||||
            configuration.isTransitive = false
 | 
			
		||||
            configuration.resolve().single()
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /** Define a dependency for illuaminate from a version number and the current operating system. */
 | 
			
		||||
    private fun illuaminateArtifact(project: Project, version: String): Dependency {
 | 
			
		||||
        val osName = System.getProperty("os.name").toLowerCase()
 | 
			
		||||
        val (os, suffix) = when {
 | 
			
		||||
            osName.contains("windows") -> Pair("windows", ".exe")
 | 
			
		||||
            osName.contains("mac os") || osName.contains("darwin") -> Pair("macos", "")
 | 
			
		||||
            osName.contains("linux") -> Pair("linux", "")
 | 
			
		||||
            else -> error("Unsupported OS $osName for illuaminate")
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        val osArch = System.getProperty("os.arch").toLowerCase()
 | 
			
		||||
        val arch = when {
 | 
			
		||||
            // On macOS the x86_64 binary will work for both ARM and Intel Macs through Rosetta.
 | 
			
		||||
            os == "macos" -> "x86_64"
 | 
			
		||||
            osArch == "arm" || osArch.startsWith("aarch") -> error("Unsupported architecture '$osArch' for illuaminate")
 | 
			
		||||
            osArch.contains("64") -> "x86_64"
 | 
			
		||||
            else -> error("Unsupported architecture '$osArch' for illuaminate")
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return project.dependencies.create(
 | 
			
		||||
            mapOf(
 | 
			
		||||
                "group" to "cc.squiddev",
 | 
			
		||||
                "name" to "illuaminate",
 | 
			
		||||
                "version" to version,
 | 
			
		||||
                "ext" to "$os-$arch$suffix",
 | 
			
		||||
            ),
 | 
			
		||||
        )
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
private val Task.illuaminatePath: String? // "?" needed to avoid overload ambiguity in setExecutable below.
 | 
			
		||||
    get() = project.extensions.getByType(IlluaminateExtension::class.java).file.get().absolutePath
 | 
			
		||||
 | 
			
		||||
/** Prepares illuaminate for being run. This simply requests the dependency and then marks it as executable. */
 | 
			
		||||
abstract class SetupIlluaminate : DefaultTask() {
 | 
			
		||||
    @get:Input
 | 
			
		||||
    abstract val file: Property<String>
 | 
			
		||||
 | 
			
		||||
    @TaskAction
 | 
			
		||||
    fun setExecutable() {
 | 
			
		||||
        val file = File(this.file.get())
 | 
			
		||||
        if (file.canExecute()) {
 | 
			
		||||
            didWork = false
 | 
			
		||||
            return
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        file.setExecutable(true)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    companion object {
 | 
			
		||||
        const val NAME: String = "setupIlluaminate"
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
abstract class IlluaminateExec : AbstractExecTask<IlluaminateExec>(IlluaminateExec::class.java) {
 | 
			
		||||
    init {
 | 
			
		||||
        dependsOn(SetupIlluaminate.NAME)
 | 
			
		||||
        executable = illuaminatePath
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
abstract class IlluaminateExecToDir : ExecToDir() {
 | 
			
		||||
    init {
 | 
			
		||||
        dependsOn(SetupIlluaminate.NAME)
 | 
			
		||||
        executable = illuaminatePath
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,62 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.artifacts.Dependency
 | 
			
		||||
import org.gradle.api.artifacts.MinimalExternalModuleDependency
 | 
			
		||||
import org.gradle.api.publish.maven.MavenPublication
 | 
			
		||||
import org.gradle.api.specs.Spec
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * A dependency in a POM file.
 | 
			
		||||
 */
 | 
			
		||||
data class MavenDependency(val groupId: String?, val artifactId: String?, val version: String?, val scope: String?)
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * A spec specifying which dependencies to include/exclude.
 | 
			
		||||
 */
 | 
			
		||||
class MavenDependencySpec {
 | 
			
		||||
    private val excludeSpecs = mutableListOf<Spec<MavenDependency>>()
 | 
			
		||||
 | 
			
		||||
    fun exclude(spec: Spec<MavenDependency>) {
 | 
			
		||||
        excludeSpecs.add(spec)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun exclude(dep: Dependency) {
 | 
			
		||||
        exclude {
 | 
			
		||||
            (dep.group.isNullOrEmpty() || dep.group == it.groupId) &&
 | 
			
		||||
                (dep.name.isNullOrEmpty() || dep.name == it.artifactId) &&
 | 
			
		||||
                (dep.version.isNullOrEmpty() || dep.version == it.version)
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun exclude(dep: MinimalExternalModuleDependency) {
 | 
			
		||||
        exclude {
 | 
			
		||||
            dep.module.group == it.groupId && dep.module.name == it.artifactId
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun isIncluded(dep: MavenDependency) = !excludeSpecs.any { it.isSatisfiedBy(dep) }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Configure dependencies present in this publication's POM file.
 | 
			
		||||
 *
 | 
			
		||||
 * While this approach is very ugly, it's the easiest way to handle it!
 | 
			
		||||
 */
 | 
			
		||||
fun MavenPublication.mavenDependencies(action: MavenDependencySpec.() -> Unit) {
 | 
			
		||||
    val spec = MavenDependencySpec()
 | 
			
		||||
    action(spec)
 | 
			
		||||
 | 
			
		||||
    pom.withXml {
 | 
			
		||||
        val dependencies = XmlUtil.findChild(asNode(), "dependencies") ?: return@withXml
 | 
			
		||||
        dependencies.children().map { it as groovy.util.Node }.forEach {
 | 
			
		||||
            val dep = MavenDependency(
 | 
			
		||||
                groupId = XmlUtil.findChild(it, "groupId")?.text(),
 | 
			
		||||
                artifactId = XmlUtil.findChild(it, "artifactId")?.text(),
 | 
			
		||||
                version = XmlUtil.findChild(it, "version")?.text(),
 | 
			
		||||
                scope = XmlUtil.findChild(it, "scope")?.text(),
 | 
			
		||||
            )
 | 
			
		||||
 | 
			
		||||
            if (!spec.isIncluded(dep)) it.parent().remove(it)
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,189 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.Project
 | 
			
		||||
import org.gradle.api.artifacts.Configuration
 | 
			
		||||
import org.gradle.api.artifacts.ModuleDependency
 | 
			
		||||
import org.gradle.api.artifacts.dsl.DependencyHandler
 | 
			
		||||
import org.gradle.api.attributes.Bundling
 | 
			
		||||
import org.gradle.api.attributes.Category
 | 
			
		||||
import org.gradle.api.attributes.LibraryElements
 | 
			
		||||
import org.gradle.api.attributes.Usage
 | 
			
		||||
import org.gradle.api.attributes.java.TargetJvmVersion
 | 
			
		||||
import org.gradle.api.capabilities.Capability
 | 
			
		||||
import org.gradle.api.plugins.BasePlugin
 | 
			
		||||
import org.gradle.api.plugins.JavaPluginExtension
 | 
			
		||||
import org.gradle.api.tasks.SourceSet
 | 
			
		||||
import org.gradle.api.tasks.bundling.Jar
 | 
			
		||||
import org.gradle.api.tasks.javadoc.Javadoc
 | 
			
		||||
import org.gradle.kotlin.dsl.get
 | 
			
		||||
import org.gradle.kotlin.dsl.named
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * This sets up a separate client-only source set, and extends that and the main/common source set with additional
 | 
			
		||||
 * metadata, to make it easier to consume jars downstream.
 | 
			
		||||
 */
 | 
			
		||||
class MinecraftConfigurations private constructor(private val project: Project) {
 | 
			
		||||
    private val java = project.extensions.getByType(JavaPluginExtension::class.java)
 | 
			
		||||
    private val sourceSets = java.sourceSets
 | 
			
		||||
    private val configurations = project.configurations
 | 
			
		||||
    private val objects = project.objects
 | 
			
		||||
 | 
			
		||||
    private val main = sourceSets[SourceSet.MAIN_SOURCE_SET_NAME]
 | 
			
		||||
    private val test = sourceSets[SourceSet.TEST_SOURCE_SET_NAME]
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Performs the initial setup of our configurations.
 | 
			
		||||
     */
 | 
			
		||||
    private fun setup() {
 | 
			
		||||
        // Define a client source set.
 | 
			
		||||
        val client = sourceSets.maybeCreate("client")
 | 
			
		||||
 | 
			
		||||
        // Ensure the client classpaths behave the same as the main ones.
 | 
			
		||||
        configurations.named(client.compileClasspathConfigurationName) {
 | 
			
		||||
            shouldResolveConsistentlyWith(configurations[main.compileClasspathConfigurationName])
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        configurations.named(client.runtimeClasspathConfigurationName) {
 | 
			
		||||
            shouldResolveConsistentlyWith(configurations[main.runtimeClasspathConfigurationName])
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // Set up an API configuration for clients (to ensure it's consistent with the main source set).
 | 
			
		||||
        val clientApi = configurations.maybeCreate(client.apiConfigurationName).apply {
 | 
			
		||||
            isVisible = false
 | 
			
		||||
            isCanBeConsumed = false
 | 
			
		||||
            isCanBeResolved = false
 | 
			
		||||
        }
 | 
			
		||||
        configurations.named(client.implementationConfigurationName) { extendsFrom(clientApi) }
 | 
			
		||||
 | 
			
		||||
        /*
 | 
			
		||||
          Now add outgoing variants for the main and common source sets that we can consume downstream. This is possibly
 | 
			
		||||
          the worst way to do things, but unfortunately the alternatives don't actually work very well:
 | 
			
		||||
 | 
			
		||||
           - Just using source set outputs: This means dependencies don't propagate, which means when :fabric depends
 | 
			
		||||
             on :fabric-api, we don't inherit the fake :common-api in IDEA.
 | 
			
		||||
 | 
			
		||||
           - Having separate common/main jars: Nice in principle, but unfortunately Forge needs a separate deobf jar
 | 
			
		||||
             task (as the original jar is obfuscated), and IDEA is not able to map its output back to a source set.
 | 
			
		||||
 | 
			
		||||
          This works for now, but is incredibly brittle. It's part of the reason we can't use testFixtures inside our
 | 
			
		||||
          MC projects, as that adds a project(self) -> test dependency, which would pull in the jar instead.
 | 
			
		||||
 | 
			
		||||
          Note we register a fake client jar here. It's not actually needed, but is there to make sure IDEA has
 | 
			
		||||
          a way to tell that client classes are needed at runtime.
 | 
			
		||||
 | 
			
		||||
          I'm so sorry, deeply aware how cursed this is.
 | 
			
		||||
        */
 | 
			
		||||
        setupOutgoing(main, "CommonOnly")
 | 
			
		||||
        project.tasks.register(client.jarTaskName, Jar::class.java) {
 | 
			
		||||
            description = "An empty jar standing in for the client classes."
 | 
			
		||||
            group = BasePlugin.BUILD_GROUP
 | 
			
		||||
            archiveClassifier.set("client")
 | 
			
		||||
        }
 | 
			
		||||
        setupOutgoing(client)
 | 
			
		||||
 | 
			
		||||
        // Reset the client classpath (Loom configures it slightly differently to this) and add a main -> client
 | 
			
		||||
        // dependency. Here we /can/ use source set outputs as we add transitive deps by patching the classpath. Nasty,
 | 
			
		||||
        // but avoids accidentally pulling in Forge's obfuscated jar.
 | 
			
		||||
        client.compileClasspath = client.compileClasspath + main.compileClasspath
 | 
			
		||||
        client.runtimeClasspath = client.runtimeClasspath + main.runtimeClasspath
 | 
			
		||||
        project.dependencies.add(client.apiConfigurationName, main.output)
 | 
			
		||||
 | 
			
		||||
        // Also add client classes to the test classpath. We do the same nasty tricks as needed for main -> client.
 | 
			
		||||
        test.compileClasspath += client.compileClasspath
 | 
			
		||||
        test.runtimeClasspath += client.runtimeClasspath
 | 
			
		||||
        project.dependencies.add(test.implementationConfigurationName, client.output)
 | 
			
		||||
 | 
			
		||||
        // Configure some tasks to include our additional files.
 | 
			
		||||
        project.tasks.named("javadoc", Javadoc::class.java) {
 | 
			
		||||
            source(client.allJava)
 | 
			
		||||
            classpath = main.compileClasspath + main.output + client.compileClasspath + client.output
 | 
			
		||||
        }
 | 
			
		||||
        // This are already done by Fabric, but we need it for Forge and vanilla. It shouldn't conflict at all.
 | 
			
		||||
        project.tasks.named("jar", Jar::class.java) { from(client.output) }
 | 
			
		||||
        project.tasks.named("sourcesJar", Jar::class.java) { from(client.allSource) }
 | 
			
		||||
 | 
			
		||||
        project.extensions.configure(CCTweakedExtension::class.java) {
 | 
			
		||||
            sourceDirectories.add(SourceSetReference.internal(client))
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun setupOutgoing(sourceSet: SourceSet, suffix: String = "") {
 | 
			
		||||
        setupOutgoing("${sourceSet.apiElementsConfigurationName}$suffix", sourceSet, objects.named(Usage.JAVA_API)) {
 | 
			
		||||
            description = "API elements for ${sourceSet.name}"
 | 
			
		||||
            extendsFrom(configurations[sourceSet.apiConfigurationName])
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        setupOutgoing("${sourceSet.runtimeElementsConfigurationName}$suffix", sourceSet, objects.named(Usage.JAVA_RUNTIME)) {
 | 
			
		||||
            description = "Runtime elements for ${sourceSet.name}"
 | 
			
		||||
            extendsFrom(configurations[sourceSet.implementationConfigurationName], configurations[sourceSet.runtimeOnlyConfigurationName])
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Set up an outgoing configuration for a specific source set. We set an additional "main" or "client" capability
 | 
			
		||||
     * (depending on the source set name) which allows downstream projects to consume them separately (see
 | 
			
		||||
     * [DependencyHandler.commonClasses] and [DependencyHandler.clientClasses]).
 | 
			
		||||
     */
 | 
			
		||||
    private fun setupOutgoing(name: String, sourceSet: SourceSet, usage: Usage, configure: Configuration.() -> Unit) {
 | 
			
		||||
        configurations.register(name) {
 | 
			
		||||
            isVisible = false
 | 
			
		||||
            isCanBeConsumed = true
 | 
			
		||||
            isCanBeResolved = false
 | 
			
		||||
 | 
			
		||||
            configure(this)
 | 
			
		||||
 | 
			
		||||
            attributes {
 | 
			
		||||
                attribute(Category.CATEGORY_ATTRIBUTE, objects.named(Category.LIBRARY))
 | 
			
		||||
                attribute(Usage.USAGE_ATTRIBUTE, usage)
 | 
			
		||||
                attribute(Bundling.BUNDLING_ATTRIBUTE, objects.named(Bundling.EXTERNAL))
 | 
			
		||||
                attributeProvider(
 | 
			
		||||
                    TargetJvmVersion.TARGET_JVM_VERSION_ATTRIBUTE,
 | 
			
		||||
                    java.toolchain.languageVersion.map { it.asInt() },
 | 
			
		||||
                )
 | 
			
		||||
                attribute(LibraryElements.LIBRARY_ELEMENTS_ATTRIBUTE, objects.named(LibraryElements.JAR))
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            outgoing {
 | 
			
		||||
                capability(BasicOutgoingCapability(project, sourceSet.name))
 | 
			
		||||
 | 
			
		||||
                // We have two outgoing variants here: the original jar and the classes.
 | 
			
		||||
                artifact(project.tasks.named(sourceSet.jarTaskName))
 | 
			
		||||
 | 
			
		||||
                variants.create("classes") {
 | 
			
		||||
                    attributes.attribute(LibraryElements.LIBRARY_ELEMENTS_ATTRIBUTE, objects.named(LibraryElements.CLASSES))
 | 
			
		||||
                    sourceSet.output.classesDirs.forEach { artifact(it) { builtBy(sourceSet.output) } }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    companion object {
 | 
			
		||||
        fun setup(project: Project) {
 | 
			
		||||
            MinecraftConfigurations(project).setup()
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
private class BasicIncomingCapability(private val module: ModuleDependency, private val name: String) : Capability {
 | 
			
		||||
    override fun getGroup(): String = module.group!!
 | 
			
		||||
    override fun getName(): String = "${module.name}-$name"
 | 
			
		||||
    override fun getVersion(): String? = null
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
private class BasicOutgoingCapability(private val project: Project, private val name: String) : Capability {
 | 
			
		||||
    override fun getGroup(): String = project.group.toString()
 | 
			
		||||
    override fun getName(): String = "${project.name}-$name"
 | 
			
		||||
    override fun getVersion(): String = project.version.toString()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fun DependencyHandler.clientClasses(notation: Any): ModuleDependency {
 | 
			
		||||
    val dep = create(notation) as ModuleDependency
 | 
			
		||||
    dep.capabilities { requireCapability(BasicIncomingCapability(dep, "client")) }
 | 
			
		||||
    return dep
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fun DependencyHandler.commonClasses(notation: Any): ModuleDependency {
 | 
			
		||||
    val dep = create(notation) as ModuleDependency
 | 
			
		||||
    dep.capabilities { requireCapability(BasicIncomingCapability(dep, "main")) }
 | 
			
		||||
    return dep
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										191
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/MinecraftExec.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										191
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/MinecraftExec.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,191 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.GradleException
 | 
			
		||||
import org.gradle.api.file.FileSystemOperations
 | 
			
		||||
import org.gradle.api.invocation.Gradle
 | 
			
		||||
import org.gradle.api.provider.Provider
 | 
			
		||||
import org.gradle.api.services.BuildService
 | 
			
		||||
import org.gradle.api.services.BuildServiceParameters
 | 
			
		||||
import org.gradle.api.tasks.*
 | 
			
		||||
import org.gradle.kotlin.dsl.getByName
 | 
			
		||||
import org.gradle.language.base.plugins.LifecycleBasePlugin
 | 
			
		||||
import java.io.File
 | 
			
		||||
import java.nio.file.Files
 | 
			
		||||
import java.util.concurrent.TimeUnit
 | 
			
		||||
import java.util.function.Supplier
 | 
			
		||||
import javax.inject.Inject
 | 
			
		||||
import kotlin.random.Random
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * A [JavaExec] task for client-tests. This sets some common setup, and uses [MinecraftRunnerService] to ensure only one
 | 
			
		||||
 * test runs at once.
 | 
			
		||||
 */
 | 
			
		||||
abstract class ClientJavaExec : JavaExec() {
 | 
			
		||||
    private val clientRunner: Provider<MinecraftRunnerService> = MinecraftRunnerService.get(project.gradle)
 | 
			
		||||
 | 
			
		||||
    init {
 | 
			
		||||
        group = LifecycleBasePlugin.VERIFICATION_GROUP
 | 
			
		||||
        usesService(clientRunner)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * When [false], tests will not be run automatically, allowing the user to debug rendering.
 | 
			
		||||
     */
 | 
			
		||||
    @get:Input
 | 
			
		||||
    val clientDebug get() = project.hasProperty("clientDebug")
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * When [false], tests will not run under a framebuffer.
 | 
			
		||||
     */
 | 
			
		||||
    @get:Input
 | 
			
		||||
    val useFramebuffer get() = !clientDebug && !project.hasProperty("clientNoFramebuffer")
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * The path test results are written to.
 | 
			
		||||
     */
 | 
			
		||||
    @get:OutputFile
 | 
			
		||||
    val testResults = project.layout.buildDirectory.file("test-results/$name.xml")
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Copy configuration from a task with the given name.
 | 
			
		||||
     */
 | 
			
		||||
    fun copyFrom(path: String) = copyFrom(project.tasks.getByName(path, JavaExec::class))
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Copy configuration from an existing [JavaExec] task.
 | 
			
		||||
     */
 | 
			
		||||
    fun copyFrom(task: JavaExec) {
 | 
			
		||||
        for (dep in task.dependsOn) dependsOn(dep)
 | 
			
		||||
        task.copyToFull(this)
 | 
			
		||||
 | 
			
		||||
        if (!clientDebug) systemProperty("cctest.client", "")
 | 
			
		||||
        systemProperty("cctest.gametest-report", testResults.get().asFile.absoluteFile)
 | 
			
		||||
        workingDir(project.buildDir.resolve("gametest").resolve(name))
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Only run tests with the given tags.
 | 
			
		||||
     */
 | 
			
		||||
    fun tags(vararg tags: String) {
 | 
			
		||||
        systemProperty("cctest.tags", tags.joinToString(","))
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Write a file with the given contents before starting Minecraft. This may be useful for writing config files.
 | 
			
		||||
     */
 | 
			
		||||
    fun withFileContents(path: Any, contents: Supplier<String>) {
 | 
			
		||||
        val file = project.file(path).toPath()
 | 
			
		||||
        doFirst {
 | 
			
		||||
            Files.createDirectories(file.parent)
 | 
			
		||||
            Files.writeString(file, contents.get())
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Copy a file to the provided path before starting Minecraft. This copy only occurs if the file does not already
 | 
			
		||||
     * exist.
 | 
			
		||||
     */
 | 
			
		||||
    fun withFileFrom(path: Any, source: Supplier<File>) {
 | 
			
		||||
        val file = project.file(path).toPath()
 | 
			
		||||
        doFirst {
 | 
			
		||||
            Files.createDirectories(file.parent)
 | 
			
		||||
            if (!Files.exists(file)) Files.copy(source.get().toPath(), file)
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @TaskAction
 | 
			
		||||
    override fun exec() {
 | 
			
		||||
        Files.createDirectories(workingDir.toPath())
 | 
			
		||||
        fsOperations.delete { delete(workingDir.resolve("screenshots")) }
 | 
			
		||||
 | 
			
		||||
        if (useFramebuffer) {
 | 
			
		||||
            clientRunner.get().wrapClient(this) { super.exec() }
 | 
			
		||||
        } else {
 | 
			
		||||
            super.exec()
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @get:Inject
 | 
			
		||||
    protected abstract val fsOperations: FileSystemOperations
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * A service for [JavaExec] tasks which start Minecraft.
 | 
			
		||||
 *
 | 
			
		||||
 * Tasks may run `usesService(MinecraftRunnerService.get(gradle))` to ensure that only one Minecraft-related task runs
 | 
			
		||||
 * at once.
 | 
			
		||||
 */
 | 
			
		||||
abstract class MinecraftRunnerService : BuildService<BuildServiceParameters.None> {
 | 
			
		||||
    private val hasXvfb = lazy {
 | 
			
		||||
        System.getProperty("os.name", "").equals("linux", ignoreCase = true) && ProcessHelpers.onPath("xvfb-run")
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    internal fun wrapClient(exec: JavaExec, run: () -> Unit) = when {
 | 
			
		||||
        hasXvfb.value -> runXvfb(exec, run)
 | 
			
		||||
        else -> run()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Run a program under Xvfb, preventing it spawning a window.
 | 
			
		||||
     */
 | 
			
		||||
    private fun runXvfb(exec: JavaExec, run: () -> Unit) {
 | 
			
		||||
        fun ProcessBuilder.startVerbose(): Process {
 | 
			
		||||
            exec.logger.info("Running ${this.command()}")
 | 
			
		||||
            return start()
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        CloseScope().use { scope ->
 | 
			
		||||
            val dir = Files.createTempDirectory("cctweaked").toAbsolutePath()
 | 
			
		||||
            scope.add { fsOperations.delete { delete(dir) } }
 | 
			
		||||
 | 
			
		||||
            val authFile = Files.createTempFile(dir, "Xauthority", "").toAbsolutePath()
 | 
			
		||||
 | 
			
		||||
            val cookie = StringBuilder().also {
 | 
			
		||||
                for (i in 0..31) it.append("0123456789abcdef"[Random.nextInt(16)])
 | 
			
		||||
            }.toString()
 | 
			
		||||
 | 
			
		||||
            val xvfb =
 | 
			
		||||
                ProcessBuilder("Xvfb", "-displayfd", "1", "-screen", "0", "640x480x24", "-nolisten", "tcp").also {
 | 
			
		||||
                    it.inheritIO()
 | 
			
		||||
                    it.environment()["XAUTHORITY"] = authFile.toString()
 | 
			
		||||
                    it.redirectOutput(ProcessBuilder.Redirect.PIPE)
 | 
			
		||||
                }.startVerbose()
 | 
			
		||||
            scope.add { xvfb.destroyForcibly().waitFor() }
 | 
			
		||||
 | 
			
		||||
            val server = xvfb.inputReader().use { it.readLine().trim() }
 | 
			
		||||
            exec.logger.info("Running at :$server (XAUTHORITY=$authFile.toA")
 | 
			
		||||
 | 
			
		||||
            ProcessBuilder("xauth", "add", ":$server", ".", cookie).also {
 | 
			
		||||
                it.inheritIO()
 | 
			
		||||
                it.environment()["XAUTHORITY"] = authFile.toString()
 | 
			
		||||
            }.startVerbose().waitForOrThrow("Failed to setup XAuthority file")
 | 
			
		||||
 | 
			
		||||
            scope.add {
 | 
			
		||||
                ProcessBuilder("xauth", "remove", ":$server").also {
 | 
			
		||||
                    it.inheritIO()
 | 
			
		||||
                    it.environment()["XAUTHORITY"] = authFile.toString()
 | 
			
		||||
                }.startVerbose().waitFor()
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            // Wait a few seconds for Xvfb to start. Ugly, but identical to xvfb-run.
 | 
			
		||||
            if (xvfb.waitFor(3, TimeUnit.SECONDS)) {
 | 
			
		||||
                throw GradleException("Xvfb unexpectedly exited (with status code ${xvfb.exitValue()})")
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            exec.environment("XAUTHORITY", authFile.toString())
 | 
			
		||||
            exec.environment("DISPLAY", ":$server")
 | 
			
		||||
 | 
			
		||||
            run()
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @get:Inject
 | 
			
		||||
    protected abstract val fsOperations: FileSystemOperations
 | 
			
		||||
 | 
			
		||||
    companion object {
 | 
			
		||||
        fun get(gradle: Gradle): Provider<MinecraftRunnerService> =
 | 
			
		||||
            gradle.sharedServices.registerIfAbsent("cc.tweaked.gradle.ClientJavaExec", MinecraftRunnerService::class.java) {
 | 
			
		||||
                maxParallelUsages.set(1)
 | 
			
		||||
            }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										60
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/Node.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/Node.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,60 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.DefaultTask
 | 
			
		||||
import org.gradle.api.Plugin
 | 
			
		||||
import org.gradle.api.Project
 | 
			
		||||
import org.gradle.api.file.Directory
 | 
			
		||||
import org.gradle.api.file.DirectoryProperty
 | 
			
		||||
import org.gradle.api.provider.Provider
 | 
			
		||||
import org.gradle.api.tasks.*
 | 
			
		||||
import java.io.File
 | 
			
		||||
 | 
			
		||||
class NodePlugin : Plugin<Project> {
 | 
			
		||||
    override fun apply(project: Project) {
 | 
			
		||||
        val extension = project.extensions.create("node", NodeExtension::class.java)
 | 
			
		||||
        project.tasks.register(NpmInstall.TASK_NAME, NpmInstall::class.java) {
 | 
			
		||||
            projectRoot.convention(extension.projectRoot)
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
abstract class NodeExtension(project: Project) {
 | 
			
		||||
    /** The directory containing `package-lock.json` and `node_modules/`. */
 | 
			
		||||
    abstract val projectRoot: DirectoryProperty
 | 
			
		||||
 | 
			
		||||
    init {
 | 
			
		||||
        projectRoot.convention(project.layout.projectDirectory)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/** Installs node modules as dependencies. */
 | 
			
		||||
abstract class NpmInstall : DefaultTask() {
 | 
			
		||||
    @get:Internal
 | 
			
		||||
    abstract val projectRoot: DirectoryProperty
 | 
			
		||||
 | 
			
		||||
    @get:InputFile
 | 
			
		||||
    @get:PathSensitive(PathSensitivity.NONE)
 | 
			
		||||
    val packageLock: Provider<File> = projectRoot.file("package-lock.json").map { it.asFile }
 | 
			
		||||
 | 
			
		||||
    @get:OutputDirectory
 | 
			
		||||
    val nodeModules: Provider<Directory> = projectRoot.dir("node_modules")
 | 
			
		||||
 | 
			
		||||
    @TaskAction
 | 
			
		||||
    fun install() {
 | 
			
		||||
        project.exec {
 | 
			
		||||
            commandLine("npm", "ci")
 | 
			
		||||
            workingDir = projectRoot.get().asFile
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    companion object {
 | 
			
		||||
        internal const val TASK_NAME: String = "npmInstall"
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
abstract class NpxExecToDir : ExecToDir() {
 | 
			
		||||
    init {
 | 
			
		||||
        dependsOn(NpmInstall.TASK_NAME)
 | 
			
		||||
        executable = "npx"
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										50
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/ProcessHelpers.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/ProcessHelpers.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,50 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.codehaus.groovy.runtime.ProcessGroovyMethods
 | 
			
		||||
import org.gradle.api.GradleException
 | 
			
		||||
import java.io.BufferedReader
 | 
			
		||||
import java.io.File
 | 
			
		||||
import java.io.InputStreamReader
 | 
			
		||||
 | 
			
		||||
internal object ProcessHelpers {
 | 
			
		||||
    fun startProcess(vararg command: String): Process {
 | 
			
		||||
        // Something randomly passes in "GIT_DIR=" as an environment variable which clobbers everything else. Don't
 | 
			
		||||
        // inherit the environment array!
 | 
			
		||||
        return ProcessBuilder()
 | 
			
		||||
            .command(*command)
 | 
			
		||||
            .redirectError(ProcessBuilder.Redirect.INHERIT)
 | 
			
		||||
            .also { it.environment().clear() }
 | 
			
		||||
            .start()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun captureOut(vararg command: String): String {
 | 
			
		||||
        val process = startProcess(*command)
 | 
			
		||||
        process.outputStream.close()
 | 
			
		||||
 | 
			
		||||
        val result = ProcessGroovyMethods.getText(process)
 | 
			
		||||
        process.waitForOrThrow("Failed to run command")
 | 
			
		||||
        return result
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun captureLines(vararg command: String): List<String> {
 | 
			
		||||
        val process = startProcess(*command)
 | 
			
		||||
        process.outputStream.close()
 | 
			
		||||
 | 
			
		||||
        val out = BufferedReader(InputStreamReader(process.inputStream)).use { reader ->
 | 
			
		||||
            reader.lines().filter { it.isNotEmpty() }.toList()
 | 
			
		||||
        }
 | 
			
		||||
        ProcessGroovyMethods.closeStreams(process)
 | 
			
		||||
        process.waitForOrThrow("Failed to run command")
 | 
			
		||||
        return out
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fun onPath(name: String): Boolean {
 | 
			
		||||
        val path = System.getenv("PATH") ?: return false
 | 
			
		||||
        return path.splitToSequence(File.pathSeparator).any { File(it, name).exists() }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
internal fun Process.waitForOrThrow(message: String) {
 | 
			
		||||
    val ret = waitFor()
 | 
			
		||||
    if (ret != 0) throw GradleException("$message (exited with $ret)")
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,20 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import org.gradle.api.tasks.SourceSet
 | 
			
		||||
 | 
			
		||||
data class SourceSetReference(
 | 
			
		||||
    val sourceSet: SourceSet,
 | 
			
		||||
    val classes: Boolean,
 | 
			
		||||
    val external: Boolean,
 | 
			
		||||
) {
 | 
			
		||||
    companion object {
 | 
			
		||||
        /** A source set in the current project. */
 | 
			
		||||
        fun internal(sourceSet: SourceSet) = SourceSetReference(sourceSet, classes = true, external = false)
 | 
			
		||||
 | 
			
		||||
        /** A source set from another project. */
 | 
			
		||||
        fun external(sourceSet: SourceSet) = SourceSetReference(sourceSet, classes = true, external = true)
 | 
			
		||||
 | 
			
		||||
        /** A source set which is inlined into the current project. */
 | 
			
		||||
        fun inline(sourceSet: SourceSet) = SourceSetReference(sourceSet, classes = false, external = false)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										12
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/XmlUtil.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								buildSrc/src/main/kotlin/cc/tweaked/gradle/XmlUtil.kt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,12 @@
 | 
			
		||||
package cc.tweaked.gradle
 | 
			
		||||
 | 
			
		||||
import groovy.util.Node
 | 
			
		||||
import groovy.util.NodeList
 | 
			
		||||
 | 
			
		||||
object XmlUtil {
 | 
			
		||||
    fun findChild(node: Node, name: String): Node? = when (val child = node.get(name)) {
 | 
			
		||||
        is Node -> child
 | 
			
		||||
        is NodeList -> child.singleOrNull() as Node?
 | 
			
		||||
        else -> null
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,12 +0,0 @@
 | 
			
		||||
#!/bin/sh
 | 
			
		||||
cd luaj-2.0.3
 | 
			
		||||
echo "Building LuaJ..."
 | 
			
		||||
ant clean
 | 
			
		||||
ant
 | 
			
		||||
 | 
			
		||||
echo "Copying output to libs..."
 | 
			
		||||
rm ../libs/luaj-jse-2.0.3.jar
 | 
			
		||||
cp luaj-jse-2.0.3.jar ../libs
 | 
			
		||||
 | 
			
		||||
echo "Done."
 | 
			
		||||
cd ..
 | 
			
		||||
							
								
								
									
										10
									
								
								codesize.sh
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								codesize.sh
									
									
									
									
									
								
							@@ -1,10 +0,0 @@
 | 
			
		||||
#!/bin/sh
 | 
			
		||||
echo "Java code:"
 | 
			
		||||
cat `find src | grep \\.java$` | wc
 | 
			
		||||
 | 
			
		||||
echo "Lua code:"
 | 
			
		||||
cat `find src/main/resources/assets/computercraft/lua | grep \\.lua$` | wc
 | 
			
		||||
 | 
			
		||||
echo "JSON:"
 | 
			
		||||
cat `find src/main/resources/assets/computercraft | grep \\.json$` | wc
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										167
									
								
								config/checkstyle/checkstyle.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										167
									
								
								config/checkstyle/checkstyle.xml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,167 @@
 | 
			
		||||
<?xml version="1.0" encoding="UTF-8"?>
 | 
			
		||||
<!DOCTYPE module PUBLIC
 | 
			
		||||
    "-//Checkstyle//DTD Checkstyle Configuration 1.3//EN"
 | 
			
		||||
    "https://checkstyle.org/dtds/configuration_1_3.dtd">
 | 
			
		||||
<module name="Checker">
 | 
			
		||||
    <property name="tabWidth" value="4"/>
 | 
			
		||||
    <property name="charset" value="UTF-8" />
 | 
			
		||||
 | 
			
		||||
    <module name="SuppressionFilter">
 | 
			
		||||
	<property name="file" value="${config_loc}/suppressions.xml" />
 | 
			
		||||
    </module>
 | 
			
		||||
 | 
			
		||||
    <module name="BeforeExecutionExclusionFileFilter">
 | 
			
		||||
        <property name="fileNamePattern" value="render_old"/>
 | 
			
		||||
    </module>
 | 
			
		||||
 | 
			
		||||
    <module name="TreeWalker">
 | 
			
		||||
        <!-- Annotations -->
 | 
			
		||||
        <module name="AnnotationLocation" />
 | 
			
		||||
        <module name="AnnotationUseStyle">
 | 
			
		||||
            <!-- We want trailing commas on multiline arrays. -->
 | 
			
		||||
            <property name="trailingArrayComma" value="ignore" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="MissingDeprecated" />
 | 
			
		||||
        <module name="MissingOverride" />
 | 
			
		||||
 | 
			
		||||
        <!-- Blocks -->
 | 
			
		||||
        <module name="EmptyBlock" />
 | 
			
		||||
        <module name="EmptyCatchBlock">
 | 
			
		||||
            <property name="exceptionVariableName" value="ignored" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="LeftCurly" />
 | 
			
		||||
        <module name="NeedBraces">
 | 
			
		||||
            <property name="allowSingleLineStatement" value="true"/>
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="RightCurly" />
 | 
			
		||||
 | 
			
		||||
        <!-- Class design. As if we've ever followed good practice here. -->
 | 
			
		||||
        <module name="FinalClass" />
 | 
			
		||||
        <module name="InterfaceIsType" />
 | 
			
		||||
        <module name="MutableException" />
 | 
			
		||||
        <module name="OneTopLevelClass" />
 | 
			
		||||
 | 
			
		||||
        <!-- Coding -->
 | 
			
		||||
        <module name="ArrayTrailingComma" />
 | 
			
		||||
        <module name="EqualsHashCode" />
 | 
			
		||||
        <!-- FallThrough does not handle unreachable code well -->
 | 
			
		||||
        <module name="IllegalInstantiation" />
 | 
			
		||||
        <module name="IllegalThrows" />
 | 
			
		||||
        <module name="ModifiedControlVariable" />
 | 
			
		||||
        <module name="NoClone" />
 | 
			
		||||
        <module name="NoFinalizer" />
 | 
			
		||||
        <module name="OneStatementPerLine" />
 | 
			
		||||
        <module name="PackageDeclaration" />
 | 
			
		||||
        <module name="SimplifyBooleanExpression" />
 | 
			
		||||
        <module name="SimplifyBooleanReturn" />
 | 
			
		||||
        <module name="StringLiteralEquality" />
 | 
			
		||||
        <module name="UnnecessaryParentheses">
 | 
			
		||||
            <!-- Default minus LAND. -->
 | 
			
		||||
            <property name="tokens" value="EXPR,IDENT,NUM_DOUBLE,NUM_FLOAT,NUM_INT,NUM_LONG,STRING_LITERAL,LITERAL_NULL,LITERAL_FALSE,LITERAL_TRUE,ASSIGN,BAND_ASSIGN,BOR_ASSIGN,BSR_ASSIGN,BXOR_ASSIGN,DIV_ASSIGN,MINUS_ASSIGN,MOD_ASSIGN,PLUS_ASSIGN,SL_ASSIGN,SR_ASSIGN,STAR_ASSIGN,LAMBDA,TEXT_BLOCK_LITERAL_BEGIN,LITERAL_INSTANCEOF,GT,LT,GE,LE,EQUAL,NOT_EQUAL,UNARY_MINUS,UNARY_PLUS,INC,DEC,LNOT,BNOT,POST_INC,POST_DEC" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="UnnecessarySemicolonAfterTypeMemberDeclaration" />
 | 
			
		||||
        <module name="UnnecessarySemicolonInTryWithResources" />
 | 
			
		||||
        <module name="UnnecessarySemicolonInEnumeration" />
 | 
			
		||||
 | 
			
		||||
        <!-- Imports -->
 | 
			
		||||
        <module name="CustomImportOrder">
 | 
			
		||||
            <property name="customImportOrderRules"
 | 
			
		||||
                value="THIRD_PARTY_PACKAGE###STANDARD_JAVA_PACKAGE###STATIC"
 | 
			
		||||
            />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="IllegalImport" />
 | 
			
		||||
        <module name="RedundantImport" />
 | 
			
		||||
        <module name="UnusedImports" />
 | 
			
		||||
 | 
			
		||||
        <!-- Javadoc -->
 | 
			
		||||
        <!-- TODO: Missing* checks for the dan200.computercraft.api package? -->
 | 
			
		||||
        <module name="AtclauseOrder">
 | 
			
		||||
            <property name="tagOrder" value="@param, @return, @throws, @deprecated"/>
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="InvalidJavadocPosition" />
 | 
			
		||||
        <module name="JavadocBlockTagLocation" />
 | 
			
		||||
        <module name="JavadocMethod"/>
 | 
			
		||||
        <module name="JavadocType"/>
 | 
			
		||||
        <module name="JavadocStyle">
 | 
			
		||||
            <property name="checkHtml" value="false" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="NonEmptyAtclauseDescription" />
 | 
			
		||||
        <module name="SingleLineJavadoc" />
 | 
			
		||||
        <module name="SummaryJavadocCheck"/>
 | 
			
		||||
 | 
			
		||||
        <!-- Misc -->
 | 
			
		||||
        <module name="ArrayTypeStyle" />
 | 
			
		||||
        <module name="CommentsIndentation" />
 | 
			
		||||
        <module name="Indentation" />
 | 
			
		||||
        <module name="OuterTypeFilename" />
 | 
			
		||||
 | 
			
		||||
        <!-- Modifiers -->
 | 
			
		||||
        <module name="ModifierOrder" />
 | 
			
		||||
        <module name="RedundantModifier" />
 | 
			
		||||
 | 
			
		||||
        <!-- Naming -->
 | 
			
		||||
        <module name="ClassTypeParameterName" />
 | 
			
		||||
        <module name="InterfaceTypeParameterName" />
 | 
			
		||||
        <module name="LambdaParameterName" />
 | 
			
		||||
        <module name="LocalFinalVariableName" />
 | 
			
		||||
        <module name="LocalVariableName" />
 | 
			
		||||
        <module name="MemberName" />
 | 
			
		||||
        <module name="MethodName">
 | 
			
		||||
            <property name="format" value="^(computercraft\$)?[a-z][a-zA-Z0-9]*$" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="MethodTypeParameterName" />
 | 
			
		||||
        <module name="PackageName">
 | 
			
		||||
            <property name="format" value="^(dan200\.computercraft|cc\.tweaked)(\.[a-z][a-z0-9]*)*" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="ParameterName" />
 | 
			
		||||
        <module name="StaticVariableName">
 | 
			
		||||
            <property name="format" value="^[a-z][a-zA-Z0-9]*|CAPABILITY(_[A-Z_]+)?$" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="TypeName" />
 | 
			
		||||
 | 
			
		||||
        <!-- Whitespace -->
 | 
			
		||||
        <module name="EmptyForInitializerPad"/>
 | 
			
		||||
        <module name="EmptyForIteratorPad">
 | 
			
		||||
            <property name="option" value="space"/>
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="GenericWhitespace" />
 | 
			
		||||
        <module name="MethodParamPad" />
 | 
			
		||||
        <module name="NoLineWrap" />
 | 
			
		||||
        <module name="NoWhitespaceAfter">
 | 
			
		||||
            <property name="tokens" value="AT,INC,DEC,UNARY_MINUS,UNARY_PLUS,BNOT,LNOT,DOT,ARRAY_DECLARATOR,INDEX_OP,METHOD_REF" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="NoWhitespaceBefore" />
 | 
			
		||||
        <!-- TODO: Decide on an OperatorWrap style. -->
 | 
			
		||||
        <module name="ParenPad" />
 | 
			
		||||
        <module name="SeparatorWrap">
 | 
			
		||||
            <property name="option" value="eol" />
 | 
			
		||||
            <property name="tokens" value="COMMA,SEMI,ELLIPSIS,ARRAY_DECLARATOR,RBRACK,METHOD_REF" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="SeparatorWrap">
 | 
			
		||||
            <property name="option" value="nl" />
 | 
			
		||||
            <property name="tokens" value="DOT,AT" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="SingleSpaceSeparator" />
 | 
			
		||||
        <module name="TypecastParenPad" />
 | 
			
		||||
        <module name="WhitespaceAfter">
 | 
			
		||||
            <property name="tokens" value="COMMA" />
 | 
			
		||||
        </module>
 | 
			
		||||
        <module name="WhitespaceAround">
 | 
			
		||||
            <property name="ignoreEnhancedForColon" value="false" />
 | 
			
		||||
            <!-- Allow empty functions -->
 | 
			
		||||
            <property name="allowEmptyLambdas" value="true" />
 | 
			
		||||
            <property name="allowEmptyMethods" value="true" />
 | 
			
		||||
            <property name="allowEmptyConstructors" value="true" />
 | 
			
		||||
            <property name="allowEmptyTypes" value="true" />
 | 
			
		||||
 | 
			
		||||
            <property name="tokens" value="ASSIGN,BAND,BAND_ASSIGN,BOR,BOR_ASSIGN,BSR,BSR_ASSIGN,BXOR,BXOR_ASSIGN,COLON,DIV,DIV_ASSIGN,EQUAL,GE,GT,LAMBDA,LAND,LCURLY,LE,LITERAL_RETURN,LOR,LT,MINUS,MINUS_ASSIGN,MOD,MOD_ASSIGN,NOT_EQUAL,PLUS,PLUS_ASSIGN,QUESTION,RCURLY,SL,SLIST,SL_ASSIGN,SR,SR_ASSIGN,STAR,STAR_ASSIGN,LITERAL_ASSERT,TYPE_EXTENSION_AND" />
 | 
			
		||||
        </module>
 | 
			
		||||
    </module>
 | 
			
		||||
 | 
			
		||||
    <module name="FileTabCharacter" />
 | 
			
		||||
    <module name="NewlineAtEndOfFile" />
 | 
			
		||||
    <module name="RegexpSingleline">
 | 
			
		||||
        <property name="format" value="\s+$"/>
 | 
			
		||||
        <property name="message" value="Trailing whitespace"/>
 | 
			
		||||
    </module>
 | 
			
		||||
</module>
 | 
			
		||||
							
								
								
									
										12
									
								
								config/checkstyle/suppressions.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								config/checkstyle/suppressions.xml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,12 @@
 | 
			
		||||
<?xml version="1.0" encoding="UTF-8"?>
 | 
			
		||||
<!DOCTYPE suppressions PUBLIC
 | 
			
		||||
    "-//Checkstyle//DTD SuppressionFilter Configuration 1.2//EN"
 | 
			
		||||
    "https://checkstyle.org/dtds/suppressions_1_2.dtd">
 | 
			
		||||
<suppressions>
 | 
			
		||||
    <!-- All the config options and method fields. -->
 | 
			
		||||
    <suppress checks="StaticVariableName" files=".*[\\/]ComputerCraft.java" />
 | 
			
		||||
    <suppress checks="StaticVariableName" files=".*[\\/]ComputerCraftAPI.java" />
 | 
			
		||||
 | 
			
		||||
    <!-- The commands API is documented in Lua. -->
 | 
			
		||||
    <suppress checks="SummaryJavadocCheck" files=".*[\\/]CommandAPI.java" />
 | 
			
		||||
</suppressions>
 | 
			
		||||
							
								
								
									
										8
									
								
								config/gitpod/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								config/gitpod/Dockerfile
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,8 @@
 | 
			
		||||
FROM gitpod/workspace-base
 | 
			
		||||
 | 
			
		||||
USER gitpod
 | 
			
		||||
 | 
			
		||||
RUN sudo apt-get -q update \
 | 
			
		||||
 && sudo apt-get install -yq openjdk-16-jdk python3-pip npm \
 | 
			
		||||
 && sudo pip3 install pre-commit \
 | 
			
		||||
 && sudo update-java-alternatives --set java-1.16.0-openjdk-amd64
 | 
			
		||||
							
								
								
									
										3
									
								
								config/license/api.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								config/license/api.txt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
			
		||||
This file is part of the public ComputerCraft API - http://www.computercraft.info
 | 
			
		||||
Copyright Daniel Ratcliffe, 2011-${year}. This API may be redistributed unmodified and in full only.
 | 
			
		||||
For help using the API, and posting your mods, visit the forums at computercraft.info.
 | 
			
		||||
							
								
								
									
										3
									
								
								config/license/main.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								config/license/main.txt
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
			
		||||
This file is part of ComputerCraft - http://www.computercraft.info
 | 
			
		||||
Copyright Daniel Ratcliffe, 2011-${year}. Do not distribute without permission.
 | 
			
		||||
Send enquiries to dratcliffe@gmail.com
 | 
			
		||||
							
								
								
									
										21
									
								
								deploy.sh
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								deploy.sh
									
									
									
									
									
								
							@@ -1,21 +0,0 @@
 | 
			
		||||
#!/bin/sh
 | 
			
		||||
 | 
			
		||||
echo "Building with gradle..."
 | 
			
		||||
rm -rf build/libs
 | 
			
		||||
rm -rf build/resources
 | 
			
		||||
rm -rf build/classes
 | 
			
		||||
chmod -R +rw src/main/resources
 | 
			
		||||
chmod +x gradlew
 | 
			
		||||
./gradlew build
 | 
			
		||||
 | 
			
		||||
echo "Deleting old deployment..."
 | 
			
		||||
rm -rf deploy
 | 
			
		||||
mkdir deploy
 | 
			
		||||
 | 
			
		||||
echo "Making new deployment..."
 | 
			
		||||
INPUTJAR=`ls -1 build/libs | grep -v sources`
 | 
			
		||||
OUTPUTJAR=`ls -1 build/libs | grep -v sources | sed s/\-//g`
 | 
			
		||||
FRIENDLYNAME=`ls -1 build/libs | grep -v sources | sed s/\-/\ /g | sed s/\.jar//g`
 | 
			
		||||
cp build/libs/$INPUTJAR deploy/$OUTPUTJAR
 | 
			
		||||
 | 
			
		||||
echo "Done."
 | 
			
		||||
							
								
								
									
										21
									
								
								doc/events/alarm.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								doc/events/alarm.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] alarm
 | 
			
		||||
see: os.setAlarm To start an alarm.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{timer} event is fired when an alarm started with @{os.setAlarm} completes.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The ID of the alarm that finished.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Starts a timer and then prints its ID:
 | 
			
		||||
```lua
 | 
			
		||||
local alarmID = os.setAlarm(os.time() + 0.05)
 | 
			
		||||
local event, id
 | 
			
		||||
repeat
 | 
			
		||||
    event, id = os.pullEvent("alarm")
 | 
			
		||||
until id == alarmID
 | 
			
		||||
print("Alarm with ID " .. id .. " was fired")
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										24
									
								
								doc/events/char.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								doc/events/char.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,24 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] char
 | 
			
		||||
see: key To listen to any key press.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{char} event is fired when a character is _typed_ on the keyboard.
 | 
			
		||||
 | 
			
		||||
The @{char} event is different to a key press. Sometimes multiple key presses may result in one character being
 | 
			
		||||
typed (for instance, on some European keyboards). Similarly, some keys (e.g. <kbd>Ctrl</kbd>) do not have any
 | 
			
		||||
corresponding character. The @{key} should be used if you want to listen to key presses themselves.
 | 
			
		||||
 | 
			
		||||
## Return values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The string representing the character that was pressed.
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints each character the user presses:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, character = os.pullEvent("char")
 | 
			
		||||
  print(character .. " was pressed.")
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										18
									
								
								doc/events/computer_command.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								doc/events/computer_command.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] computer_command
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{computer_command} event is fired when the `/computercraft queue` command is run for the current computer.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
... @{string}: The arguments passed to the command.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints the contents of messages sent:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event = {os.pullEvent("computer_command")}
 | 
			
		||||
  print("Received message:", table.unpack(event, 2))
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										19
									
								
								doc/events/disk.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								doc/events/disk.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] disk
 | 
			
		||||
see: disk_eject For the event sent when a disk is removed.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{disk} event is fired when a disk is inserted into an adjacent or networked disk drive.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The side of the disk drive that had a disk inserted.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when a disk is inserted:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, side = os.pullEvent("disk")
 | 
			
		||||
  print("Inserted a disk on side " .. side)
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										19
									
								
								doc/events/disk_eject.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								doc/events/disk_eject.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] disk_eject
 | 
			
		||||
see: disk For the event sent when a disk is inserted.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{disk_eject} event is fired when a disk is removed from an adjacent or networked disk drive.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The side of the disk drive that had a disk removed.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when a disk is removed:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, side = os.pullEvent("disk_eject")
 | 
			
		||||
  print("Removed a disk on side " .. side)
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										42
									
								
								doc/events/file_transfer.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								doc/events/file_transfer.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,42 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] file_transfer
 | 
			
		||||
since: 1.101.0
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{file_transfer} event is queued when a user drags-and-drops a file on an open computer.
 | 
			
		||||
 | 
			
		||||
This event contains a single argument, that in turn has a single method @{TransferredFiles.getFiles|getFiles}. This
 | 
			
		||||
returns the list of files that are being transferred. Each file is a @{fs.BinaryReadHandle|binary file handle} with an
 | 
			
		||||
additional @{TransferredFile.getName|getName} method.
 | 
			
		||||
 | 
			
		||||
## Return values
 | 
			
		||||
1. @{string}: The event name
 | 
			
		||||
2. @{TransferredFiles}: The list of transferred files.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Waits for a user to drop files on top of the computer, then prints the list of files and the size of each file.
 | 
			
		||||
 | 
			
		||||
```lua
 | 
			
		||||
local _, files = os.pullEvent("file_transfer")
 | 
			
		||||
for _, file in ipairs(files.getFiles()) do
 | 
			
		||||
  -- Seek to the end of the file to get its size, then go back to the beginning.
 | 
			
		||||
  local size = file.seek("end")
 | 
			
		||||
  file.seek("set", 0)
 | 
			
		||||
 | 
			
		||||
  print(file.getName() .. " " .. file.getSize())
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Save each transferred file to the computer's storage.
 | 
			
		||||
 | 
			
		||||
```lua
 | 
			
		||||
local _, files = os.pullEvent("file_transfer")
 | 
			
		||||
for _, file in ipairs(files.getFiles()) do
 | 
			
		||||
  local handle = fs.open(file.getName(), "wb")
 | 
			
		||||
  handle.write(file.readAll())
 | 
			
		||||
 | 
			
		||||
  handle.close()
 | 
			
		||||
  file.close()
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										14
									
								
								doc/events/http_check.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								doc/events/http_check.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,14 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] http_check
 | 
			
		||||
see: http.checkURLAsync To check a URL asynchronously.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{http_check} event is fired when a URL check finishes.
 | 
			
		||||
 | 
			
		||||
This event is normally handled inside @{http.checkURL}, but it can still be seen when using @{http.checkURLAsync}.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The URL requested to be checked.
 | 
			
		||||
3. @{boolean}: Whether the check succeeded.
 | 
			
		||||
4. @{string|nil}: If the check failed, a reason explaining why the check failed.
 | 
			
		||||
							
								
								
									
										39
									
								
								doc/events/http_failure.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								doc/events/http_failure.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] http_failure
 | 
			
		||||
see: http.request To send an HTTP request.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{http_failure} event is fired when an HTTP request fails.
 | 
			
		||||
 | 
			
		||||
This event is normally handled inside @{http.get} and @{http.post}, but it can still be seen when using @{http.request}.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The URL of the site requested.
 | 
			
		||||
3. @{string}: An error describing the failure.
 | 
			
		||||
4. @{http.Response|nil}: A response handle if the connection succeeded, but the server's response indicated failure.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints an error why the website cannot be contacted:
 | 
			
		||||
```lua
 | 
			
		||||
local myURL = "https://does.not.exist.tweaked.cc"
 | 
			
		||||
http.request(myURL)
 | 
			
		||||
local event, url, err
 | 
			
		||||
repeat
 | 
			
		||||
    event, url, err = os.pullEvent("http_failure")
 | 
			
		||||
until url == myURL
 | 
			
		||||
print("The URL " .. url .. " could not be reached: " .. err)
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Prints the contents of a webpage that does not exist:
 | 
			
		||||
```lua
 | 
			
		||||
local myURL = "https://tweaked.cc/this/does/not/exist"
 | 
			
		||||
http.request(myURL)
 | 
			
		||||
local event, url, err, handle
 | 
			
		||||
repeat
 | 
			
		||||
    event, url, err, handle = os.pullEvent("http_failure")
 | 
			
		||||
until url == myURL
 | 
			
		||||
print("The URL " .. url .. " could not be reached: " .. err)
 | 
			
		||||
print(handle.getResponseCode())
 | 
			
		||||
handle.close()
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										27
									
								
								doc/events/http_success.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								doc/events/http_success.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] http_success
 | 
			
		||||
see: http.request To make an HTTP request.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{http_success} event is fired when an HTTP request returns successfully.
 | 
			
		||||
 | 
			
		||||
This event is normally handled inside @{http.get} and @{http.post}, but it can still be seen when using @{http.request}.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The URL of the site requested.
 | 
			
		||||
3. @{http.Response}: The handle for the response text.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints the content of a website (this may fail if the request fails):
 | 
			
		||||
```lua
 | 
			
		||||
local myURL = "https://tweaked.cc/"
 | 
			
		||||
http.request(myURL)
 | 
			
		||||
local event, url, handle
 | 
			
		||||
repeat
 | 
			
		||||
    event, url, handle = os.pullEvent("http_success")
 | 
			
		||||
until url == myURL
 | 
			
		||||
print("Contents of " .. url .. ":")
 | 
			
		||||
print(handle.readAll())
 | 
			
		||||
handle.close()
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										26
									
								
								doc/events/key.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								doc/events/key.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,26 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] key
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
This event is fired when any key is pressed while the terminal is focused.
 | 
			
		||||
 | 
			
		||||
This event returns a numerical "key code" (for instance, <kbd>F1</kbd> is 290). This value may vary between versions and
 | 
			
		||||
so it is recommended to use the constants in the @{keys} API rather than hard coding numeric values.
 | 
			
		||||
 | 
			
		||||
If the button pressed represented a printable character, then the @{key} event will be followed immediately by a @{char}
 | 
			
		||||
event. If you are consuming text input, use a @{char} event instead!
 | 
			
		||||
 | 
			
		||||
## Return values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The numerical key value of the key pressed.
 | 
			
		||||
3. @{boolean}: Whether the key event was generated while holding the key (@{true}), rather than pressing it the first time (@{false}).
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints each key when the user presses it, and if the key is being held.
 | 
			
		||||
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, key, is_held = os.pullEvent("key")
 | 
			
		||||
  print(("%s held=%s"):format(keys.getName(key), is_held))
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										24
									
								
								doc/events/key_up.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								doc/events/key_up.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,24 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] key_up
 | 
			
		||||
see: keys For a lookup table of the given keys.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
Fired whenever a key is released (or the terminal is closed while a key was being pressed).
 | 
			
		||||
 | 
			
		||||
This event returns a numerical "key code" (for instance, <kbd>F1</kbd> is 290). This value may vary between versions and
 | 
			
		||||
so it is recommended to use the constants in the @{keys} API rather than hard coding numeric values.
 | 
			
		||||
 | 
			
		||||
## Return values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The numerical key value of the key pressed.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints each key released on the keyboard whenever a @{key_up} event is fired.
 | 
			
		||||
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, key = os.pullEvent("key_up")
 | 
			
		||||
  local name = keys.getName(key) or "unknown key"
 | 
			
		||||
  print(name .. " was released.")
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										26
									
								
								doc/events/modem_message.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								doc/events/modem_message.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,26 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] modem_message
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{modem_message} event is fired when a message is received on an open channel on any @{modem}.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The side of the modem that received the message.
 | 
			
		||||
3. @{number}: The channel that the message was sent on.
 | 
			
		||||
4. @{number}: The reply channel set by the sender.
 | 
			
		||||
5. @{any}: The message as sent by the sender.
 | 
			
		||||
6. @{number}: The distance between the sender and the receiver, in blocks.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Wraps a @{modem} peripheral, opens channel 0 for listening, and prints all received messages.
 | 
			
		||||
 | 
			
		||||
```lua
 | 
			
		||||
local modem = peripheral.find("modem") or error("No modem attached", 0)
 | 
			
		||||
modem.open(0)
 | 
			
		||||
 | 
			
		||||
while true do
 | 
			
		||||
  local event, side, channel, replyChannel, message, distance = os.pullEvent("modem_message")
 | 
			
		||||
  print(("Message received on side %s on channel %d (reply to %d) from %f blocks away with message %s"):format(side, channel, replyChannel, distance, tostring(message)))
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										18
									
								
								doc/events/monitor_resize.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								doc/events/monitor_resize.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] monitor_resize
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{monitor_resize} event is fired when an adjacent or networked monitor's size is changed.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The side or network ID of the monitor that resized.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when a monitor is resized:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, side = os.pullEvent("monitor_resize")
 | 
			
		||||
  print("The monitor on side " .. side .. " was resized.")
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										20
									
								
								doc/events/monitor_touch.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								doc/events/monitor_touch.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,20 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] monitor_touch
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{monitor_touch} event is fired when an adjacent or networked Advanced Monitor is right-clicked.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The side or network ID of the monitor that was touched.
 | 
			
		||||
3. @{number}: The X coordinate of the touch, in characters.
 | 
			
		||||
4. @{number}: The Y coordinate of the touch, in characters.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when a monitor is touched:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, side, x, y = os.pullEvent("monitor_touch")
 | 
			
		||||
  print("The monitor on side " .. side .. " was touched at (" .. x .. ", " .. y .. ")")
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										32
									
								
								doc/events/mouse_click.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								doc/events/mouse_click.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,32 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] mouse_click
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
This event is fired when the terminal is clicked with a mouse. This event is only fired on advanced computers (including
 | 
			
		||||
advanced turtles and pocket computers).
 | 
			
		||||
 | 
			
		||||
## Return values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The mouse button that was clicked.
 | 
			
		||||
3. @{number}: The X-coordinate of the click.
 | 
			
		||||
4. @{number}: The Y-coordinate of the click.
 | 
			
		||||
 | 
			
		||||
## Mouse buttons
 | 
			
		||||
Several mouse events (@{mouse_click}, @{mouse_up}, @{mouse_scroll}) contain a "mouse button" code. This takes a
 | 
			
		||||
numerical value depending on which button on your mouse was last pressed when this event occurred.
 | 
			
		||||
 | 
			
		||||
| Button Code | Mouse Button  |
 | 
			
		||||
|------------:|---------------|
 | 
			
		||||
|           1 | Left button   |
 | 
			
		||||
|           2 | Right button  |
 | 
			
		||||
|           3 | Middle button |
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Print the button and the coordinates whenever the mouse is clicked.
 | 
			
		||||
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, button, x, y = os.pullEvent("mouse_click")
 | 
			
		||||
  print(("The mouse button %s was pressed at %d, %d"):format(button, x, y))
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										22
									
								
								doc/events/mouse_drag.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								doc/events/mouse_drag.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,22 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] mouse_drag
 | 
			
		||||
see: mouse_click For when a mouse button is initially pressed.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
This event is fired every time the mouse is moved while a mouse button is being held.
 | 
			
		||||
 | 
			
		||||
## Return values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The [mouse button](mouse_click.html#Mouse_buttons) that is being pressed.
 | 
			
		||||
3. @{number}: The X-coordinate of the mouse.
 | 
			
		||||
4. @{number}: The Y-coordinate of the mouse.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Print the button and the coordinates whenever the mouse is dragged.
 | 
			
		||||
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, button, x, y = os.pullEvent("mouse_drag")
 | 
			
		||||
  print(("The mouse button %s was dragged at %d, %d"):format(button, x, y))
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										21
									
								
								doc/events/mouse_scroll.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								doc/events/mouse_scroll.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] mouse_scroll
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
This event is fired when a mouse wheel is scrolled in the terminal.
 | 
			
		||||
 | 
			
		||||
## Return values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The direction of the scroll. (-1 = up, 1 = down)
 | 
			
		||||
3. @{number}: The X-coordinate of the mouse when scrolling.
 | 
			
		||||
4. @{number}: The Y-coordinate of the mouse when scrolling.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints the direction of each scroll, and the position of the mouse at the time.
 | 
			
		||||
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, dir, x, y = os.pullEvent("mouse_scroll")
 | 
			
		||||
  print(("The mouse was scrolled in direction %s at %d, %d"):format(dir, x, y))
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										21
									
								
								doc/events/mouse_up.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								doc/events/mouse_up.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] mouse_up
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
This event is fired when a mouse button is released or a held mouse leaves the computer's terminal.
 | 
			
		||||
 | 
			
		||||
## Return values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The [mouse button](mouse_click.html#Mouse_buttons) that was released.
 | 
			
		||||
3. @{number}: The X-coordinate of the mouse.
 | 
			
		||||
4. @{number}: The Y-coordinate of the mouse.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints the coordinates and button number whenever the mouse is released.
 | 
			
		||||
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, button, x, y = os.pullEvent("mouse_up")
 | 
			
		||||
  print(("The mouse button %s was released at %d, %d"):format(button, x, y))
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										18
									
								
								doc/events/paste.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								doc/events/paste.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] paste
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{paste} event is fired when text is pasted into the computer through Ctrl-V (or ⌘V on Mac).
 | 
			
		||||
 | 
			
		||||
## Return values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string} The text that was pasted.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints pasted text:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, text = os.pullEvent("paste")
 | 
			
		||||
  print('"' .. text .. '" was pasted')
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										19
									
								
								doc/events/peripheral.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								doc/events/peripheral.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] peripheral
 | 
			
		||||
see: peripheral_detach For the event fired when a peripheral is detached.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{peripheral} event is fired when a peripheral is attached on a side or to a modem.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The side the peripheral was attached to.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when a peripheral is attached:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, side = os.pullEvent("peripheral")
 | 
			
		||||
  print("A peripheral was attached on side " .. side)
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										19
									
								
								doc/events/peripheral_detach.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								doc/events/peripheral_detach.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] peripheral_detach
 | 
			
		||||
see: peripheral For the event fired when a peripheral is attached.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{peripheral_detach} event is fired when a peripheral is detached from a side or from a modem.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The side the peripheral was detached from.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when a peripheral is detached:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, side = os.pullEvent("peripheral_detach")
 | 
			
		||||
  print("A peripheral was detached on side " .. side)
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										30
									
								
								doc/events/rednet_message.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								doc/events/rednet_message.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,30 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] rednet_message
 | 
			
		||||
see: modem_message For raw modem messages sent outside of Rednet.
 | 
			
		||||
see: rednet.receive To wait for a Rednet message with an optional timeout and protocol filter.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{rednet_message} event is fired when a message is sent over Rednet.
 | 
			
		||||
 | 
			
		||||
This event is usually handled by @{rednet.receive}, but it can also be pulled manually.
 | 
			
		||||
 | 
			
		||||
@{rednet_message} events are sent by @{rednet.run} in the top-level coroutine in response to @{modem_message} events. A @{rednet_message} event is always preceded by a @{modem_message} event. They are generated inside CraftOS rather than being sent by the ComputerCraft machine.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The ID of the sending computer.
 | 
			
		||||
3. @{any}: The message sent.
 | 
			
		||||
4. @{string|nil}: The protocol of the message, if provided.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when one is sent:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event, sender, message, protocol = os.pullEvent("rednet_message")
 | 
			
		||||
  if protocol ~= nil then
 | 
			
		||||
    print("Received message from " .. sender .. " with protocol " .. protocol .. " and message " .. tostring(message))
 | 
			
		||||
  else
 | 
			
		||||
    print("Received message from " .. sender .. " with message " .. tostring(message))
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										14
									
								
								doc/events/redstone.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								doc/events/redstone.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,14 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] redstone
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{event!redstone} event is fired whenever any redstone inputs on the computer change.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when a redstone input changes:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  os.pullEvent("redstone")
 | 
			
		||||
  print("A redstone input has changed!")
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										27
									
								
								doc/events/speaker_audio_empty.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								doc/events/speaker_audio_empty.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] speaker_audio_empty
 | 
			
		||||
see: speaker.playAudio To play audio using the speaker
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The name of the speaker which is available to play more audio.
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
This uses @{io.lines} to read audio data in blocks of 16KiB from "example_song.dfpwm", and then attempts to play it
 | 
			
		||||
using @{speaker.playAudio}. If the speaker's buffer is full, it waits for an event and tries again.
 | 
			
		||||
 | 
			
		||||
```lua {data-peripheral=speaker}
 | 
			
		||||
local dfpwm = require("cc.audio.dfpwm")
 | 
			
		||||
local speaker = peripheral.find("speaker")
 | 
			
		||||
 | 
			
		||||
local decoder = dfpwm.make_decoder()
 | 
			
		||||
for chunk in io.lines("data/example.dfpwm", 16 * 1024) do
 | 
			
		||||
    local buffer = decoder(chunk)
 | 
			
		||||
 | 
			
		||||
    while not speaker.playAudio(buffer) do
 | 
			
		||||
        os.pullEvent("speaker_audio_empty")
 | 
			
		||||
    end
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										28
									
								
								doc/events/task_complete.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								doc/events/task_complete.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,28 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] task_complete
 | 
			
		||||
see: commands.execAsync To run a command which fires a task_complete event.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{task_complete} event is fired when an asynchronous task completes. This is usually handled inside the function call that queued the task; however, functions such as @{commands.execAsync} return immediately so the user can wait for completion.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The ID of the task that completed.
 | 
			
		||||
3. @{boolean}: Whether the command succeeded.
 | 
			
		||||
4. @{string}: If the command failed, an error message explaining the failure. (This is not present if the command succeeded.)
 | 
			
		||||
...: Any parameters returned from the command.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints the results of an asynchronous command:
 | 
			
		||||
```lua
 | 
			
		||||
local taskID = commands.execAsync("say Hello")
 | 
			
		||||
local event
 | 
			
		||||
repeat
 | 
			
		||||
    event = {os.pullEvent("task_complete")}
 | 
			
		||||
until event[2] == taskID
 | 
			
		||||
if event[3] == true then
 | 
			
		||||
  print("Task " .. event[2] .. " succeeded:", table.unpack(event, 4))
 | 
			
		||||
else
 | 
			
		||||
  print("Task " .. event[2] .. " failed: " .. event[4])
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										20
									
								
								doc/events/term_resize.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								doc/events/term_resize.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,20 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] term_resize
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{term_resize} event is fired when the main terminal is resized. For instance:
 | 
			
		||||
 - When a the tab bar is shown or hidden in @{multishell}.
 | 
			
		||||
 - When the terminal is redirected to a monitor via the "monitor" program and the monitor is resized.
 | 
			
		||||
 | 
			
		||||
When this event fires, some parts of the terminal may have been moved or deleted. Simple terminal programs (those
 | 
			
		||||
not using @{term.setCursorPos}) can ignore this event, but more complex GUI programs should redraw the entire screen.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints :
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  os.pullEvent("term_resize")
 | 
			
		||||
  local w, h = term.getSize()
 | 
			
		||||
  print("The term was resized to (" .. w .. ", " .. h .. ")")
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										25
									
								
								doc/events/terminate.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								doc/events/terminate.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,25 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] terminate
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{terminate} event is fired when <kbd>Ctrl-T</kbd> is held down.
 | 
			
		||||
 | 
			
		||||
This event is normally handled by @{os.pullEvent}, and will not be returned. However, @{os.pullEventRaw} will return this event when fired.
 | 
			
		||||
 | 
			
		||||
@{terminate} will be sent even when a filter is provided to @{os.pullEventRaw}. When using @{os.pullEventRaw} with a filter, make sure to check that the event is not @{terminate}.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when Ctrl-T is held:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  local event = os.pullEventRaw("terminate")
 | 
			
		||||
  if event == "terminate" then print("Terminate requested!") end
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Exits when Ctrl-T is held:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  os.pullEvent()
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										21
									
								
								doc/events/timer.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								doc/events/timer.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] timer
 | 
			
		||||
see: os.startTimer To start a timer.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{timer} event is fired when a timer started with @{os.startTimer} completes.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{number}: The ID of the timer that finished.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Starts a timer and then prints its ID:
 | 
			
		||||
```lua
 | 
			
		||||
local timerID = os.startTimer(2)
 | 
			
		||||
local event, id
 | 
			
		||||
repeat
 | 
			
		||||
    event, id = os.pullEvent("timer")
 | 
			
		||||
until id == timerID
 | 
			
		||||
print("Timer with ID " .. id .. " was fired")
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										14
									
								
								doc/events/turtle_inventory.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								doc/events/turtle_inventory.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,14 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] turtle_inventory
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{turtle_inventory} event is fired when a turtle's inventory is changed.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when the inventory is changed:
 | 
			
		||||
```lua
 | 
			
		||||
while true do
 | 
			
		||||
  os.pullEvent("turtle_inventory")
 | 
			
		||||
  print("The inventory was changed.")
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										21
									
								
								doc/events/websocket_closed.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								doc/events/websocket_closed.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] websocket_closed
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{websocket_closed} event is fired when an open WebSocket connection is closed.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The URL of the WebSocket that was closed.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message when a WebSocket is closed (this may take a minute):
 | 
			
		||||
```lua
 | 
			
		||||
local myURL = "wss://example.tweaked.cc/echo"
 | 
			
		||||
local ws = http.websocket(myURL)
 | 
			
		||||
local event, url
 | 
			
		||||
repeat
 | 
			
		||||
    event, url = os.pullEvent("websocket_closed")
 | 
			
		||||
until url == myURL
 | 
			
		||||
print("The WebSocket at " .. url .. " was closed.")
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										25
									
								
								doc/events/websocket_failure.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								doc/events/websocket_failure.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,25 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] websocket_failure
 | 
			
		||||
see: http.websocketAsync To send an HTTP request.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{websocket_failure} event is fired when a WebSocket connection request fails.
 | 
			
		||||
 | 
			
		||||
This event is normally handled inside @{http.websocket}, but it can still be seen when using @{http.websocketAsync}.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The URL of the site requested.
 | 
			
		||||
3. @{string}: An error describing the failure.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints an error why the website cannot be contacted:
 | 
			
		||||
```lua
 | 
			
		||||
local myURL = "wss://example.tweaked.cc/not-a-websocket"
 | 
			
		||||
http.websocketAsync(myURL)
 | 
			
		||||
local event, url, err
 | 
			
		||||
repeat
 | 
			
		||||
    event, url, err = os.pullEvent("websocket_failure")
 | 
			
		||||
until url == myURL
 | 
			
		||||
print("The URL " .. url .. " could not be reached: " .. err)
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										27
									
								
								doc/events/websocket_message.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								doc/events/websocket_message.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] websocket_message
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{websocket_message} event is fired when a message is received on an open WebSocket connection.
 | 
			
		||||
 | 
			
		||||
This event is normally handled by @{http.Websocket.receive}, but it can also be pulled manually.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The URL of the WebSocket.
 | 
			
		||||
3. @{string}: The contents of the message.
 | 
			
		||||
4. @{boolean}: Whether this is a binary message.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints a message sent by a WebSocket:
 | 
			
		||||
```lua
 | 
			
		||||
local myURL = "wss://example.tweaked.cc/echo"
 | 
			
		||||
local ws = http.websocket(myURL)
 | 
			
		||||
ws.send("Hello!")
 | 
			
		||||
local event, url, message
 | 
			
		||||
repeat
 | 
			
		||||
    event, url, message = os.pullEvent("websocket_message")
 | 
			
		||||
until url == myURL
 | 
			
		||||
print("Received message from " .. url .. " with contents " .. message)
 | 
			
		||||
ws.close()
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										28
									
								
								doc/events/websocket_success.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								doc/events/websocket_success.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,28 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=event] websocket_success
 | 
			
		||||
see: http.websocketAsync To open a WebSocket asynchronously.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
The @{websocket_success} event is fired when a WebSocket connection request returns successfully.
 | 
			
		||||
 | 
			
		||||
This event is normally handled inside @{http.websocket}, but it can still be seen when using @{http.websocketAsync}.
 | 
			
		||||
 | 
			
		||||
## Return Values
 | 
			
		||||
1. @{string}: The event name.
 | 
			
		||||
2. @{string}: The URL of the site.
 | 
			
		||||
3. @{http.Websocket}: The handle for the WebSocket.
 | 
			
		||||
 | 
			
		||||
## Example
 | 
			
		||||
Prints the content of a website (this may fail if the request fails):
 | 
			
		||||
```lua
 | 
			
		||||
local myURL = "wss://example.tweaked.cc/echo"
 | 
			
		||||
http.websocketAsync(myURL)
 | 
			
		||||
local event, url, handle
 | 
			
		||||
repeat
 | 
			
		||||
    event, url, handle = os.pullEvent("websocket_success")
 | 
			
		||||
until url == myURL
 | 
			
		||||
print("Connected to " .. url)
 | 
			
		||||
handle.send("Hello!")
 | 
			
		||||
print(handle.receive())
 | 
			
		||||
handle.close()
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										90
									
								
								doc/guides/gps_setup.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								doc/guides/gps_setup.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,90 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=guide] gps_setup
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
# Setting up GPS
 | 
			
		||||
The @{gps} API allows computers and turtles to find their current position using wireless modems.
 | 
			
		||||
 | 
			
		||||
In order to use GPS, you'll need to set up multiple *GPS hosts*. These are computers running the special `gps host`
 | 
			
		||||
program, which tell other computers the host's position. Several hosts running together are known as a *GPS
 | 
			
		||||
constellation*.
 | 
			
		||||
 | 
			
		||||
In order to give the best results, a GPS constellation needs at least four computers. More than four GPS hosts per
 | 
			
		||||
constellation is redundant, but it does not cause problems.
 | 
			
		||||
 | 
			
		||||
## Building a GPS constellation
 | 
			
		||||
{.big-image}
 | 
			
		||||
 | 
			
		||||
We are going to build our GPS constellation as shown in the image above. You will need 4 computers and either 4 wireless
 | 
			
		||||
modems or 4 ender modems. Try not to mix ender and wireless modems together as you might get some odd behavior when your
 | 
			
		||||
requesting computers are out of range.
 | 
			
		||||
 | 
			
		||||
:::tip Ender modems vs wireless modems
 | 
			
		||||
Ender modems have a very large range, which makes them very useful for setting up GPS hosts. If you do this then you
 | 
			
		||||
will likely only need one GPS constellation for the whole dimension (such as the Overworld or Nether).
 | 
			
		||||
 | 
			
		||||
If you do use wireless modems then you may find that you need multiple GPS constellations to cover your needs.
 | 
			
		||||
 | 
			
		||||
A computer needs a wireless or ender modem and to be in range of a GPS constellation that is in the same dimension as it
 | 
			
		||||
to use the GPS API. The reason for this is that ComputerCraft mimics real-life GPS by making use of the distance
 | 
			
		||||
parameter of @{modem_message|modem messages} and some maths.
 | 
			
		||||
:::
 | 
			
		||||
 | 
			
		||||
Locate where you want to place your GPS constellation. You will need an area at least 6 blocks high, 6 blocks wide, and
 | 
			
		||||
6 blocks deep (6x6x6). If you are using wireless modems then you may want to build your constellation as high as you can
 | 
			
		||||
because high altitude boosts modem message range and thus the radius that your constellation covers.
 | 
			
		||||
 | 
			
		||||
The GPS constellation will only work when it is in a loaded chunk. If you want your constellation to always be
 | 
			
		||||
accessible, you may want to permanently load the chunk using a vanilla or modded chunk loader. Make sure that your 6x6x6
 | 
			
		||||
area fits in a single chunk to reduce the number of chunks that need to be kept loaded.
 | 
			
		||||
 | 
			
		||||
Let's get started building the constellation! Place your first computer in one of the corners of your 6x6x6. Remember
 | 
			
		||||
which computer this is as other computers need to be placed relative to it. Place the second computer 4 blocks above the
 | 
			
		||||
first. Go back to your first computer and place your third computer 5 blocks in front of your first computer, leaving 4
 | 
			
		||||
blocks of air between them. Finally for the fourth computer, go back to your first computer and place it 5 blocks right
 | 
			
		||||
of your first computer, leaving 4 blocks of air between them.
 | 
			
		||||
 | 
			
		||||
With all four computers placed within the 6x6x6, place one modem on top of each computer. You should have 4 modems and 4
 | 
			
		||||
computers all within your 6x6x6 where each modem is attached to a computer and each computer has a modem.
 | 
			
		||||
 | 
			
		||||
Currently your GPS constellation will not work, that's because each host is not aware that it's a GPS host. We will fix
 | 
			
		||||
this in the next section.
 | 
			
		||||
 | 
			
		||||
## Configuring the constellation
 | 
			
		||||
Now that the structure of your constellation is built, we need to configure each host in it.
 | 
			
		||||
 | 
			
		||||
Go back to the first computer that you placed and create a startup file, by running `edit startup`.
 | 
			
		||||
 | 
			
		||||
Type the following code into the file:
 | 
			
		||||
```lua
 | 
			
		||||
shell.run("gps", "host", x, y, z)
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Escape from the computer GUI and then press <kbd>F3</kbd> to open Minecraft's debug screen and then look at the computer
 | 
			
		||||
(without opening the GUI). On the right of the screen about halfway down you should see an entry labeled `Targeted
 | 
			
		||||
Block`, the numbers correspond to the position of the block that you are looking at. Replace `x` with the first number,
 | 
			
		||||
`y` with the second number, and `z` with the third number.
 | 
			
		||||
 | 
			
		||||
For example, if I had a computer at x = 59, y = 5, z = -150, then my <kbd>F3</kbd> debug screen entry would be `Target
 | 
			
		||||
Block: 59, 5, -150` and I would change my startup file to this `shell.run("gps", "host", 59, 5, -150)`.
 | 
			
		||||
 | 
			
		||||
To hide Minecraft's debug screen, press <kbd>F3</kbd> again.
 | 
			
		||||
 | 
			
		||||
Create similar startup files for the other computers in your constellation, making sure to input the each computer's own
 | 
			
		||||
coordinates.
 | 
			
		||||
 | 
			
		||||
:::caution Modem messages come from the computer's position, not the modem's
 | 
			
		||||
Wireless modems transmit from the block that they are attached to *not* the block space that they occupy, the
 | 
			
		||||
coordinates that you input into your GPS host should be the position of the computer and not the position of the modem.
 | 
			
		||||
:::
 | 
			
		||||
 | 
			
		||||
Congratulations, your constellation is now fully set up! You can test it by placing another computer close by, placing a
 | 
			
		||||
wireless modem on it, and running the `gps locate` program (or calling the @{gps.locate} function).
 | 
			
		||||
 | 
			
		||||
:::info Why use Minecraft's coordinates?
 | 
			
		||||
CC doesn't care if you use Minecraft's coordinate system, so long as all of the GPS hosts with overlapping ranges use
 | 
			
		||||
the same reference point (requesting computers will get confused if hosts have different reference points). However,
 | 
			
		||||
using MC's coordinate system does provide a nice standard to adopt server-wide. It also is consistent with how command
 | 
			
		||||
computers get their location, they use MC's command system to get their block which returns that in MC's coordinate
 | 
			
		||||
system.
 | 
			
		||||
:::
 | 
			
		||||
							
								
								
									
										99
									
								
								doc/guides/local_ips.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								doc/guides/local_ips.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,99 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=guide] local_ips
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
# Allowing access to local IPs
 | 
			
		||||
By default, ComputerCraft blocks access to local IP addresses for security. This means you can't normally access any
 | 
			
		||||
HTTP server running on your computer. However, this may be useful for testing programs without having a remote
 | 
			
		||||
server. You can unblock these IPs in the ComputerCraft config.
 | 
			
		||||
 | 
			
		||||
 - [Minecraft 1.13 and later, CC:T 1.87.0 and later](#cc-1.87.0)
 | 
			
		||||
 - [Minecraft 1.13 and later, CC:T 1.86.2 and earlier](#cc-1.86.2)
 | 
			
		||||
 - [Minecraft 1.12.2 and earlier](#mc-1.12)
 | 
			
		||||
 | 
			
		||||
## Minecraft 1.13 and later, CC:T 1.87.0 and later {#cc-1.87.0}
 | 
			
		||||
The configuration file can be located at `serverconfig/computercraft-server.toml` inside the world folder on either
 | 
			
		||||
single-player or multiplayer. Look for lines that look like this:
 | 
			
		||||
 | 
			
		||||
```toml
 | 
			
		||||
#A list of rules which control behaviour of the "http" API for specific domains or IPs.
 | 
			
		||||
#Each rule is an item with a 'host' to match against, and a series of properties. The host may be a domain name ("pastebin.com"),
 | 
			
		||||
#wildcard ("*.pastebin.com") or CIDR notation ("127.0.0.0/8"). If no rules, the domain is blocked.
 | 
			
		||||
[[http.rules]]
 | 
			
		||||
    host = "$private"
 | 
			
		||||
    action = "deny"
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
On 1.95.0 and later, this will be a single entry with `host = "$private"`. On earlier versions, this will be a number of
 | 
			
		||||
`[[http.rules]]` with various IP addresses. You will want to remove all of the `[[http.rules]]` entires that have
 | 
			
		||||
`action = "deny"`. Then save the file and relaunch Minecraft (Server).
 | 
			
		||||
 | 
			
		||||
Here's what it should look like after removing:
 | 
			
		||||
 | 
			
		||||
```toml
 | 
			
		||||
#A list of rules which control behaviour of the "http" API for specific domains or IPs.
 | 
			
		||||
#Each rule is an item with a 'host' to match against, and a series of properties. The host may be a domain name ("pastebin.com"),
 | 
			
		||||
#wildcard ("*.pastebin.com") or CIDR notation ("127.0.0.0/8"). If no rules, the domain is blocked.
 | 
			
		||||
[[http.rules]]
 | 
			
		||||
    #The maximum size (in bytes) that a computer can send or receive in one websocket packet.
 | 
			
		||||
    max_websocket_message = 131072
 | 
			
		||||
    host = "*"
 | 
			
		||||
    #The maximum size (in bytes) that a computer can upload in a single request. This includes headers and POST text.
 | 
			
		||||
    max_upload = 4194304
 | 
			
		||||
    action = "allow"
 | 
			
		||||
    #The maximum size (in bytes) that a computer can download in a single request. Note that responses may receive more data than allowed, but this data will not be returned to the client.
 | 
			
		||||
    max_download = 16777216
 | 
			
		||||
    #The period of time (in milliseconds) to wait before a HTTP request times out. Set to 0 for unlimited.
 | 
			
		||||
    timeout = 30000
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Minecraft 1.13 and later, CC:T 1.86.2 and earlier {#cc-1.86.2}
 | 
			
		||||
The configuration file for singleplayer is at `.minecraft/config/computercraft-common.toml`. Look for lines that look
 | 
			
		||||
like this:
 | 
			
		||||
 | 
			
		||||
```toml
 | 
			
		||||
#A list of wildcards for domains or IP ranges that cannot be accessed through the "http" API on Computers.
 | 
			
		||||
#If this is empty then all whitelisted domains will be accessible. Example: "*.github.com" will block access to all subdomains of github.com.
 | 
			
		||||
#You can use domain names ("pastebin.com"), wilcards ("*.pastebin.com") or CIDR notation ("127.0.0.0/8").
 | 
			
		||||
blacklist = ["127.0.0.0/8", "10.0.0.0/8", "172.16.0.0/12", "192.168.0.0/16", "fd00::/8"]
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Remove everything inside the array, leaving the last line as `blacklist = []`. Then save the file and relaunch Minecraft.
 | 
			
		||||
 | 
			
		||||
Here's what it should look like after removing:
 | 
			
		||||
 | 
			
		||||
```toml
 | 
			
		||||
#A list of wildcards for domains or IP ranges that cannot be accessed through the "http" API on Computers.
 | 
			
		||||
#If this is empty then all whitelisted domains will be accessible. Example: "*.github.com" will block access to all subdomains of github.com.
 | 
			
		||||
#You can use domain names ("pastebin.com"), wilcards ("*.pastebin.com") or CIDR notation ("127.0.0.0/8").
 | 
			
		||||
blacklist = []
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Minecraft 1.12.2 and earlier {#mc-1.12}
 | 
			
		||||
On singleplayer, the configuration file is located at `.minecraft\config\ComputerCraft.cfg`. On multiplayer, the
 | 
			
		||||
configuration file is located at `<server folder>\config\ComputerCraft.cfg`. Look for lines that look like this:
 | 
			
		||||
 | 
			
		||||
```ini
 | 
			
		||||
# A list of wildcards for domains or IP ranges that cannot be accessed through the "http" API on Computers.
 | 
			
		||||
# If this is empty then all explicitly allowed domains will be accessible. Example: "*.github.com" will block access to all subdomains of github.com.
 | 
			
		||||
# You can use domain names ("pastebin.com"), wildcards ("*.pastebin.com") or CIDR notation ("127.0.0.0/8").
 | 
			
		||||
S:blocked_domains <
 | 
			
		||||
    127.0.0.0/8
 | 
			
		||||
    10.0.0.0/8
 | 
			
		||||
    172.16.0.0/12
 | 
			
		||||
    192.168.0.0/16
 | 
			
		||||
    fd00::/8
 | 
			
		||||
 >
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Delete everything between the `<>`, leaving the last line as `S:blocked_domains = <>`. Then save the file and relaunch
 | 
			
		||||
Minecraft (Server).
 | 
			
		||||
 | 
			
		||||
Here's what it should look like after removing:
 | 
			
		||||
 | 
			
		||||
```ini
 | 
			
		||||
# A list of wildcards for domains or IP ranges that cannot be accessed through the "http" API on Computers.
 | 
			
		||||
# If this is empty then all explicitly allowed domains will be accessible. Example: "*.github.com" will block access to all subdomains of github.com.
 | 
			
		||||
# You can use domain names ("pastebin.com"), wildcards ("*.pastebin.com") or CIDR notation ("127.0.0.0/8").
 | 
			
		||||
S:blocked_domains <>
 | 
			
		||||
```
 | 
			
		||||
							
								
								
									
										204
									
								
								doc/guides/speaker_audio.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										204
									
								
								doc/guides/speaker_audio.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,204 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=guide] speaker_audio
 | 
			
		||||
see: speaker.playAudio Play PCM audio using a speaker.
 | 
			
		||||
see: cc.audio.dfpwm Provides utilities for encoding and decoding DFPWM files.
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
# Playing audio with speakers
 | 
			
		||||
CC: Tweaked's speaker peripheral provides a powerful way to play any audio you like with the @{speaker.playAudio}
 | 
			
		||||
method. However, for people unfamiliar with digital audio, it's not the most intuitive thing to use. This guide provides
 | 
			
		||||
an introduction to digital audio, demonstrates how to play music with CC: Tweaked's speakers, and then briefly discusses
 | 
			
		||||
the more complex topic of audio processing.
 | 
			
		||||
 | 
			
		||||
## A short introduction to digital audio
 | 
			
		||||
When sound is recorded it is captured as an analogue signal, effectively the electrical version of a sound
 | 
			
		||||
wave. However, this signal is continuous, and so can't be used directly by a computer. Instead, we measure (or *sample*)
 | 
			
		||||
the amplitude of the wave many times a second and then *quantise* that amplitude, rounding it to the nearest
 | 
			
		||||
representable value.
 | 
			
		||||
 | 
			
		||||
This representation of sound - a long, uniformally sampled list of amplitudes is referred to as [Pulse-code
 | 
			
		||||
Modulation][PCM] (PCM). PCM can be thought of as the "standard" audio format, as it's incredibly easy to work with. For
 | 
			
		||||
instance, to mix two pieces of audio together, you can just add samples from the two tracks together and take the average.
 | 
			
		||||
 | 
			
		||||
CC: Tweaked's speakers also work with PCM audio. It plays back 48,000 samples a second, where each sample is an integer
 | 
			
		||||
between -128 and 127. This is more commonly referred to as 48kHz and an 8-bit resolution.
 | 
			
		||||
 | 
			
		||||
Let's now look at a quick example. We're going to generate a [Sine Wave] at 220Hz, which sounds like a low monotonous
 | 
			
		||||
hum. First we wrap our speaker peripheral, and then we fill a table (also referred to as a *buffer*) with 128×1024
 | 
			
		||||
samples - this is the maximum number of samples a speaker can accept in one go.
 | 
			
		||||
 | 
			
		||||
In order to fill this buffer, we need to do a little maths. We want to play 220 sine waves each second, where each sine
 | 
			
		||||
wave completes a full oscillation in 2π "units". This means one seconds worth of audio is 2×π×220 "units" long. We then
 | 
			
		||||
need to split this into 48k samples, basically meaning for each sample we move 2×π×220/48k "along" the sine curve.
 | 
			
		||||
 | 
			
		||||
```lua {data-peripheral=speaker}
 | 
			
		||||
local speaker = peripheral.find("speaker")
 | 
			
		||||
 | 
			
		||||
local buffer = {}
 | 
			
		||||
local t, dt = 0, 2 * math.pi * 220 / 48000
 | 
			
		||||
for i = 1, 128 * 1024 do
 | 
			
		||||
    buffer[i] = math.floor(math.sin(t) * 127)
 | 
			
		||||
    t = (t + dt) % (math.pi * 2)
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
speaker.playAudio(buffer)
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Streaming audio
 | 
			
		||||
You might notice that the above snippet only generates a short bit of audio - 2.7s seconds to be precise. While we could
 | 
			
		||||
try increasing the number of loop iterations, we'll get an error when we try to play it through the speaker: the sound
 | 
			
		||||
buffer is too large for it to handle.
 | 
			
		||||
 | 
			
		||||
Our 2.7 seconds of audio is stored in a table with over 130 _thousand_ elements. If we wanted to play a full minute of
 | 
			
		||||
sine waves (and why wouldn't you?), you'd need a table with almost 3 _million_. Suddenly you find these numbers adding
 | 
			
		||||
up very quickly, and these tables take up more and more memory.
 | 
			
		||||
 | 
			
		||||
Instead of building our entire song (well, sine wave) in one go, we can produce it in small batches, each of which get
 | 
			
		||||
passed off to @{speaker.playAudio} when the time is right. This allows us to build a _stream_ of audio, where we read
 | 
			
		||||
chunks of audio one at a time (either from a file or a tone generator like above), do some optional processing to each
 | 
			
		||||
one, and then play them.
 | 
			
		||||
 | 
			
		||||
Let's adapt our example from above to do that instead.
 | 
			
		||||
 | 
			
		||||
```lua {data-peripheral=speaker}
 | 
			
		||||
local speaker = peripheral.find("speaker")
 | 
			
		||||
 | 
			
		||||
local t, dt = 0, 2 * math.pi * 220 / 48000
 | 
			
		||||
while true do
 | 
			
		||||
    local buffer = {}
 | 
			
		||||
    for i = 1, 16 * 1024 * 8 do
 | 
			
		||||
        buffer[i] = math.floor(math.sin(t) * 127)
 | 
			
		||||
        t = (t + dt) % (math.pi * 2)
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    while not speaker.playAudio(buffer) do
 | 
			
		||||
        os.pullEvent("speaker_audio_empty")
 | 
			
		||||
    end
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
It looks pretty similar to before, aside from we've wrapped the generation and playing code in a while loop, and added a
 | 
			
		||||
rather odd loop with @{speaker.playAudio} and @{os.pullEvent}.
 | 
			
		||||
 | 
			
		||||
Let's talk about this loop, why do we need to keep calling @{speaker.playAudio}? Remember that what we're trying to do
 | 
			
		||||
here is avoid keeping too much audio in memory at once. However, if we're generating audio quicker than the speakers can
 | 
			
		||||
play it, we're not helping at all - all this audio is still hanging around waiting to be played!
 | 
			
		||||
 | 
			
		||||
In order to avoid this, the speaker rejects any new chunks of audio if its backlog is too large. When this happens,
 | 
			
		||||
@{speaker.playAudio} returns false. Once enough audio has played, and the backlog has been reduced, a
 | 
			
		||||
@{speaker_audio_empty} event is queued, and we can try to play our chunk once more.
 | 
			
		||||
 | 
			
		||||
## Storing audio
 | 
			
		||||
PCM is a fantastic way of representing audio when we want to manipulate it, but it's not very efficient when we want to
 | 
			
		||||
store it to disk. Compare the size of a WAV file (which uses PCM) to an equivalent MP3, it's often 5 times the size.
 | 
			
		||||
Instead, we store audio in special formats (or *codecs*) and then convert them to PCM when we need to do processing on
 | 
			
		||||
them.
 | 
			
		||||
 | 
			
		||||
Modern audio codecs use some incredibly impressive techniques to compress the audio as much as possible while preserving
 | 
			
		||||
sound quality. However, due to CC: Tweaked's limited processing power, it's not really possible to use these from your
 | 
			
		||||
computer. Instead, we need something much simpler.
 | 
			
		||||
 | 
			
		||||
DFPWM (Dynamic Filter Pulse Width Modulation) is the de facto standard audio format of the ComputerCraft (and
 | 
			
		||||
OpenComputers) world. Originally popularised by the addon mod [Computronics], CC:T now has built-in support for it with
 | 
			
		||||
the @{cc.audio.dfpwm} module. This allows you to read DFPWM files from disk, decode them to PCM, and then play them
 | 
			
		||||
using the speaker.
 | 
			
		||||
 | 
			
		||||
Let's dive in with an example, and we'll explain things afterwards:
 | 
			
		||||
 | 
			
		||||
```lua {data-peripheral=speaker}
 | 
			
		||||
local dfpwm = require("cc.audio.dfpwm")
 | 
			
		||||
local speaker = peripheral.find("speaker")
 | 
			
		||||
 | 
			
		||||
local decoder = dfpwm.make_decoder()
 | 
			
		||||
for chunk in io.lines("data/example.dfpwm", 16 * 1024) do
 | 
			
		||||
    local buffer = decoder(chunk)
 | 
			
		||||
 | 
			
		||||
    while not speaker.playAudio(buffer) do
 | 
			
		||||
        os.pullEvent("speaker_audio_empty")
 | 
			
		||||
    end
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Once again, we see the @{speaker.playAudio}/@{speaker_audio_empty} loop. However, the rest of the program is a little
 | 
			
		||||
different.
 | 
			
		||||
 | 
			
		||||
First, we require the dfpwm module and call @{cc.audio.dfpwm.make_decoder} to construct a new decoder. This decoder
 | 
			
		||||
accepts blocks of DFPWM data and converts it to a list of 8-bit amplitudes, which we can then play with our speaker.
 | 
			
		||||
 | 
			
		||||
As mentioned above, @{speaker.playAudio} accepts at most 128×1024 samples in one go. DFPMW uses a single bit for each
 | 
			
		||||
sample, which means we want to process our audio in chunks of 16×1024 bytes (16KiB). In order to do this, we use
 | 
			
		||||
@{io.lines}, which provides a nice way to loop over chunks of a file. You can of course just use @{fs.open} and
 | 
			
		||||
@{fs.BinaryReadHandle.read} if you prefer.
 | 
			
		||||
 | 
			
		||||
## Processing audio
 | 
			
		||||
As mentioned near the beginning of this guide, PCM audio is pretty easy to work with as it's just a list of amplitudes.
 | 
			
		||||
You can mix together samples from different streams by adding their amplitudes, change the rate of playback by removing
 | 
			
		||||
samples, etc...
 | 
			
		||||
 | 
			
		||||
Let's put together a small demonstration here. We're going to add a small delay effect to the song above, so that you
 | 
			
		||||
hear a faint echo a second and a half later.
 | 
			
		||||
 | 
			
		||||
In order to do this, we'll follow a format similar to the previous example, decoding the audio and then playing it.
 | 
			
		||||
However, we'll also add some new logic between those two steps, which loops over every sample in our chunk of audio, and
 | 
			
		||||
adds the sample from 1.5 seconds ago to it.
 | 
			
		||||
 | 
			
		||||
For this, we'll need to keep track of the last 72k samples - exactly 1.5 seconds worth of audio. We can do this using a
 | 
			
		||||
[Ring Buffer], which helps makes things a little more efficient.
 | 
			
		||||
 | 
			
		||||
```lua {data-peripheral=speaker}
 | 
			
		||||
local dfpwm = require("cc.audio.dfpwm")
 | 
			
		||||
local speaker = peripheral.find("speaker")
 | 
			
		||||
 | 
			
		||||
-- Speakers play at 48kHz, so 1.5 seconds is 72k samples. We first fill our buffer
 | 
			
		||||
-- with 0s, as there's nothing to echo at the start of the track!
 | 
			
		||||
local samples_i, samples_n = 1, 48000 * 1.5
 | 
			
		||||
local samples = {}
 | 
			
		||||
for i = 1, samples_n do samples[i] = 0 end
 | 
			
		||||
 | 
			
		||||
local decoder = dfpwm.make_decoder()
 | 
			
		||||
for chunk in io.lines("data/example.dfpwm", 16 * 1024) do
 | 
			
		||||
    local buffer = decoder(chunk)
 | 
			
		||||
 | 
			
		||||
    for i = 1, #buffer do
 | 
			
		||||
        local original_value = buffer[i]
 | 
			
		||||
 | 
			
		||||
        -- Replace this sample with its current amplitude plus the amplitude from 1.5 seconds ago.
 | 
			
		||||
        -- We scale both to ensure the resulting value is still between -128 and 127.
 | 
			
		||||
        buffer[i] = original_value * 0.6 + samples[samples_i] * 0.4
 | 
			
		||||
 | 
			
		||||
        -- Now store the current sample, and move the "head" of our ring buffer forward one place.
 | 
			
		||||
        samples[samples_i] = original_value
 | 
			
		||||
        samples_i = samples_i + 1
 | 
			
		||||
        if samples_i > samples_n then samples_i = 1 end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    while not speaker.playAudio(buffer) do
 | 
			
		||||
        os.pullEvent("speaker_audio_empty")
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    -- The audio processing above can be quite slow and preparing the first batch of audio
 | 
			
		||||
    -- may timeout the computer. We sleep to avoid this.
 | 
			
		||||
    -- There's definitely better ways of handling this - this is just an example!
 | 
			
		||||
    sleep(0.05)
 | 
			
		||||
end
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
:::note Confused?
 | 
			
		||||
Don't worry if you don't understand this example. It's quite advanced, and does use some ideas that this guide doesn't
 | 
			
		||||
cover. That said, don't be afraid to ask on [GitHub Discussions] or [IRC] either!
 | 
			
		||||
:::
 | 
			
		||||
 | 
			
		||||
It's worth noting that the examples of audio processing we've mentioned here are about manipulating the _amplitude_ of
 | 
			
		||||
the wave. If you wanted to modify the _frequency_ (for instance, shifting the pitch), things get rather more complex.
 | 
			
		||||
For this, you'd need to use the [Fast Fourier transform][FFT] to convert the stream of amplitudes to frequencies,
 | 
			
		||||
process those, and then convert them back to amplitudes.
 | 
			
		||||
 | 
			
		||||
This is, I'm afraid, left as an exercise to the reader.
 | 
			
		||||
 | 
			
		||||
[Computronics]: https://github.com/Vexatos/Computronics/ "Computronics on GitHub"
 | 
			
		||||
[FFT]: https://en.wikipedia.org/wiki/Fast_Fourier_transform "Fast Fourier transform - Wikipedia"
 | 
			
		||||
[PCM]: https://en.wikipedia.org/wiki/Pulse-code_modulation "Pulse-code Modulation - Wikipedia"
 | 
			
		||||
[Ring Buffer]: https://en.wikipedia.org/wiki/Circular_buffer "Circular buffer - Wikipedia"
 | 
			
		||||
[Sine Wave]: https://en.wikipedia.org/wiki/Sine_wave "Sine wave - Wikipedia"
 | 
			
		||||
[GitHub Discussions]: https://github.com/cc-tweaked/CC-Tweaked/discussions
 | 
			
		||||
[IRC]: https://webchat.esper.net/?channels=computercraft "#computercraft on EsperNet"
 | 
			
		||||
							
								
								
									
										83
									
								
								doc/guides/using_require.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								doc/guides/using_require.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,83 @@
 | 
			
		||||
---
 | 
			
		||||
module: [kind=guide] using_require
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
# Reusing code with require
 | 
			
		||||
A library is a collection of useful functions and other definitions which is stored separately to your main program. You
 | 
			
		||||
might want to create a library because you have some functions which are used in multiple programs, or just to split
 | 
			
		||||
your program into multiple more modular files.
 | 
			
		||||
 | 
			
		||||
Let's say we want to create a small library to make working with the @{term|terminal} a little easier. We'll provide two
 | 
			
		||||
functions: `reset`, which clears the terminal and sets the cursor to (1, 1), and `write_center`, which prints some text
 | 
			
		||||
in the middle of the screen.
 | 
			
		||||
 | 
			
		||||
Start off by creating a file called `more_term.lua`:
 | 
			
		||||
 | 
			
		||||
```lua {data-snippet=more_term}
 | 
			
		||||
local function reset()
 | 
			
		||||
  term.clear()
 | 
			
		||||
  term.setCursorPos(1, 1)
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
local function write_center(text)
 | 
			
		||||
  local x, y = term.getCursorPos()
 | 
			
		||||
  local width, height = term.getSize()
 | 
			
		||||
  term.setCursorPos(math.floor((width - #text) / 2) + 1, y)
 | 
			
		||||
  term.write(text)
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
return { reset = reset, write_center = write_center }
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Now, what's going on here? We define our two functions as one might expect, and then at the bottom return a table with
 | 
			
		||||
the two functions. When we require this library, this table is what is returned. With that, we can then call the
 | 
			
		||||
original functions. Now create a new file, with the following:
 | 
			
		||||
 | 
			
		||||
```lua {data-mount=more_term:more_term.lua}
 | 
			
		||||
local more_term = require("more_term")
 | 
			
		||||
more_term.reset()
 | 
			
		||||
more_term.write_center("Hello, world!")
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
When run, this'll clear the screen and print some text in the middle of the first line.
 | 
			
		||||
 | 
			
		||||
## require in depth
 | 
			
		||||
While the previous section is a good introduction to how @{require} operates, there are a couple of remaining points
 | 
			
		||||
which are worth mentioning for more advanced usage.
 | 
			
		||||
 | 
			
		||||
### Libraries can return anything
 | 
			
		||||
In our above example, we return a table containing the functions we want to expose. However, it's worth pointing out
 | 
			
		||||
that you can return ''anything'' from your library - a table, a function or even just a string! @{require} treats them
 | 
			
		||||
all the same, and just returns whatever your library provides.
 | 
			
		||||
 | 
			
		||||
### Module resolution and the package path
 | 
			
		||||
In the above examples, we defined our library in a file, and @{require} read from it. While this is what you'll do most
 | 
			
		||||
of the time, it is possible to make @{require} look elsewhere for your library, such as downloading from a website or
 | 
			
		||||
loading from an in-memory library store.
 | 
			
		||||
 | 
			
		||||
As a result, the *module name* you pass to @{require} doesn't correspond to a file path. One common mistake is to load
 | 
			
		||||
code from a sub-directory using `require("folder/library")` or even `require("folder/library.lua")`, neither of which
 | 
			
		||||
will do quite what you expect.
 | 
			
		||||
 | 
			
		||||
When loading libraries (also referred to as *modules*) from files, @{require} searches along the *@{package.path|module
 | 
			
		||||
path}*. By default, this looks something like:
 | 
			
		||||
 | 
			
		||||
* `?.lua`
 | 
			
		||||
* `?/init.lua`
 | 
			
		||||
* `/rom/modules/main/?.lua`
 | 
			
		||||
* etc...
 | 
			
		||||
 | 
			
		||||
When you call `require("my_library")`, @{require} replaces the `?` in each element of the path with your module name, and
 | 
			
		||||
checks if the file exists. In this case, we'd look for `my_library.lua`, `my_library/init.lua`,
 | 
			
		||||
`/rom/modules/main/my_library.lua` and so on. Note that this works *relative to the current program*, so if your
 | 
			
		||||
program is actually called `folder/program`, then we'll look for `folder/my_library.lua`, etc...
 | 
			
		||||
 | 
			
		||||
One other caveat is loading libraries from sub-directories. For instance, say we have a file
 | 
			
		||||
`my/fancy/library.lua`. This can be loaded by using `require("my.fancy.library")` - the '.'s are replaced with '/'
 | 
			
		||||
before we start looking for the library.
 | 
			
		||||
 | 
			
		||||
## External links
 | 
			
		||||
There are several external resources which go into require in a little more detail:
 | 
			
		||||
 | 
			
		||||
 - The [Lua Module tutorial](http://lua-users.org/wiki/ModulesTutorial) on the Lua wiki.
 | 
			
		||||
 - [Lua's manual section on @{require}](https://www.lua.org/manual/5.1/manual.html#pdf-require).
 | 
			
		||||
							
								
								
									
										1
									
								
								doc/head.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								doc/head.html
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
<meta name="theme-color" content="#c8d87c">
 | 
			
		||||
							
								
								
									
										
											BIN
										
									
								
								doc/images/basic-terminal.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								doc/images/basic-terminal.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 53 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								doc/images/gps-constellation-example.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								doc/images/gps-constellation-example.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 331 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								doc/images/peripherals.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								doc/images/peripherals.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 194 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								doc/images/turtle.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								doc/images/turtle.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 163 KiB  | 
							
								
								
									
										54
									
								
								doc/index.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								doc/index.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,54 @@
 | 
			
		||||
# 
 | 
			
		||||
CC: Tweaked is a mod for Minecraft which adds programmable computers, turtles and more to the game. A fork of the
 | 
			
		||||
much-beloved [ComputerCraft], it continues its legacy with better performance, stability, and a wealth of new features.
 | 
			
		||||
 | 
			
		||||
CC: Tweaked can be installed from [CurseForge] or [Modrinth]. It requires the [Minecraft Forge][forge] mod loader, but
 | 
			
		||||
[versions are available for Fabric][ccrestitched].
 | 
			
		||||
 | 
			
		||||
## Features
 | 
			
		||||
Controlled using the [Lua programming language][lua], CC: Tweaked's computers provides all the tools you need to start
 | 
			
		||||
writing code and automating your Minecraft world.
 | 
			
		||||
 | 
			
		||||
{.big-image}
 | 
			
		||||
 | 
			
		||||
While computers are incredibly powerful, they're rather limited by their inability to move about. *Turtles* are the
 | 
			
		||||
solution here. They can move about the world, placing and breaking blocks, swinging a sword to protect you from zombies,
 | 
			
		||||
or whatever else you program them to!
 | 
			
		||||
 | 
			
		||||
{.big-image}
 | 
			
		||||
 | 
			
		||||
Not all problems can be solved with a pickaxe though, and so CC: Tweaked also provides a bunch of additional peripherals
 | 
			
		||||
for your computers. You can play a tune with speakers, display text or images on a monitor, connect all your
 | 
			
		||||
computers together with modems, and much more.
 | 
			
		||||
 | 
			
		||||
Computers can now also interact with inventories such as chests, allowing you to build complex inventory and item
 | 
			
		||||
management systems.
 | 
			
		||||
 | 
			
		||||
{.big-image}
 | 
			
		||||
 | 
			
		||||
## Getting Started
 | 
			
		||||
While ComputerCraft is lovely for both experienced programmers and for people who have never coded before, it can be a
 | 
			
		||||
little daunting getting started. Thankfully, there's several fantastic tutorials out there:
 | 
			
		||||
 | 
			
		||||
 - [Direwolf20's ComputerCraft tutorials](https://www.youtube.com/watch?v=wrUHUhfCY5A "ComputerCraft Tutorial Episode 1 - HELP! and Hello World")
 | 
			
		||||
 - [Sethbling's ComputerCraft series](https://www.youtube.com/watch?v=DSsx4VSe-Uk "Programming Tutorial with Minecraft Turtles -- Ep. 1: Intro to Turtles and If-Then-Else_End")
 | 
			
		||||
 - [Lyqyd's Computer Basics 1](http://www.computercraft.info/forums2/index.php?/topic/15033-computer-basics-i/ "Computer Basics I")
 | 
			
		||||
 | 
			
		||||
Once you're a little more familiar with the mod, the sidebar and links below provide more detailed documentation on the
 | 
			
		||||
various APIs and peripherals provided by the mod.
 | 
			
		||||
 | 
			
		||||
If you get stuck, do [ask a question on GitHub][GitHub Discussions] or pop in to the ComputerCraft's [IRC channel][IRC].
 | 
			
		||||
 | 
			
		||||
## Get Involved
 | 
			
		||||
CC: Tweaked lives on [GitHub]. If you've got any ideas, feedback or bugs please do [create an issue][bug].
 | 
			
		||||
 | 
			
		||||
[github]: https://github.com/cc-tweaked/CC-Tweaked/ "CC: Tweaked on GitHub"
 | 
			
		||||
[bug]: https://github.com/cc-tweaked/CC-Tweaked/issues/new/choose
 | 
			
		||||
[computercraft]: https://github.com/dan200/ComputerCraft "ComputerCraft on GitHub"
 | 
			
		||||
[curseforge]: https://minecraft.curseforge.com/projects/cc-tweaked "Download CC: Tweaked from CurseForge"
 | 
			
		||||
[modrinth]: https://modrinth.com/mod/gu7yAYhd "Download CC: Tweaked from Modrinth"
 | 
			
		||||
[forge]: https://files.minecraftforge.net/ "Download Minecraft Forge."
 | 
			
		||||
[ccrestitched]: https://www.curseforge.com/minecraft/mc-mods/cc-restitched "Download CC: Restitched from CurseForge"
 | 
			
		||||
[lua]: https://www.lua.org/ "Lua's main website"
 | 
			
		||||
[GitHub Discussions]: https://github.com/cc-tweaked/CC-Tweaked/discussions
 | 
			
		||||
[IRC]: https://webchat.esper.net/?channels=computercraft "#computercraft on EsperNet"
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user