Compare commits
	
		
			1127 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | c26168f22e | ||
|   | 391be342ac | ||
|   | 16549f8f84 | ||
|   | 2d2b65d9fa | ||
|   | a12ad37522 | ||
|   | 84541faf92 | ||
|   | c7daaba370 | ||
|   | f128813a5e | ||
|   | 699a55ea08 | ||
|   | 90a674f99b | ||
|   | f236d36ed3 | ||
|   | 1b2200ccf4 | ||
|   | c6353f3502 | ||
|   | f09b8a78a0 | ||
|   | 5f741ac46e | ||
|   | a434cfbe0d | ||
|   | cd4c7689b7 | ||
|   | 3cc2049fb3 | ||
|   | 0d3117e472 | ||
|   | b335975c97 | ||
|   | 5511b6e013 | ||
|   | 262d5ead51 | ||
|   | a78b048720 | ||
|   | e7dd5e155d | ||
|   | 5a3ea38bbf | ||
|   | 786521eba7 | ||
|   | 202cb3048a | ||
|   | 7e642cf700 | ||
|   | 7732aabc1e | ||
|   | 1cf9f52eca | ||
|   | d1d5e10b72 | ||
|   | c3fada264f | ||
|   | 45eeb74b83 | ||
|   | b9a6e551cd | ||
|   | c610e5ed9b | ||
|   | c2c2824e50 | ||
|   | 01fab09683 | ||
|   | e103904a04 | ||
|   | 56678cbac0 | ||
|   | 7fade8939d | ||
|   | daa557f6a3 | ||
|   | 23a257ed86 | ||
|   | 05d0620491 | ||
|   | 5225139284 | ||
|   | 9b2e996cae | ||
|   | 6a561342a4 | ||
|   | 429bed2f91 | ||
|   | 200d593414 | ||
|   | 9dbe12135d | ||
|   | 318d7f2652 | ||
|   | 5a653531e2 | ||
|   | c7cc3dcdfd | ||
|   | 61f54f8f74 | ||
|   | a3b27fe3a1 | ||
|   | 380749051d | ||
|   | 81625f9fc5 | ||
|   | d6ccb1725b | ||
|   | 5ce412aeda | ||
|   | fe6d88e410 | ||
|   | a21357248f | ||
|   | 70d710c9a9 | ||
|   | 183c82fb8d | ||
|   | 62cbb92154 | ||
|   | 7d70126072 | ||
|   | 54bfffa7b9 | ||
|   | 3f5b96bf62 | ||
|   | 3d8bbedf1b | ||
|   | 23c9f6a6ca | ||
|   | 5ba8d4949d | ||
|   | a6befdd541 | ||
|   | e5409db0e8 | ||
|   | 678d610cd6 | ||
|   | 466fe9c368 | ||
|   | 13feaea7b7 | ||
|   | e52f9301fa | ||
|   | 1b58e18a6d | ||
|   | b779ff08e0 | ||
|   | 811f9c22d7 | ||
|   | c3529f0691 | ||
|   | f9f574532e | ||
|   | 92dee53dd6 | ||
|   | 5d42ee2359 | ||
|   | 7c03d37caa | ||
|   | b128b593c2 | ||
|   | 342e48ed77 | ||
|   | 052f8b265d | ||
|   | 2eedc91d74 | ||
|   | 410b9ad6bc | ||
|   | 24c6dff3e4 | ||
|   | 161db7636a | ||
|   | 796252357e | ||
|   | b0344d52e9 | ||
|   | 8e6da3a0d9 | ||
|   | 756e4eaeec | ||
|   | 3126d0730a | ||
|   | 748e9f15df | ||
|   | 7c714f5788 | ||
|   | d3c3ad839b | ||
|   | 168de3c316 | ||
|   | 9e20fc5c88 | ||
|   | 1ff5151786 | ||
|   | a56738a331 | ||
|   | 2d04a3d6d2 | ||
|   | e0a55b9100 | ||
|   | 0a57eecb3a | ||
|   | 51f98020f6 | ||
|   | 70123805e1 | ||
|   | 1448b11d00 | ||
|   | 490b81ed05 | ||
|   | 96f675abed | ||
|   | 4cd451c613 | ||
|   | 187792dfc4 | ||
|   | 3dde561fe5 | ||
|   | a75ec45172 | ||
|   | 3732ddf75f | ||
|   | b8d8097734 | ||
|   | 1092532292 | ||
|   | 8a79ba0e2b | ||
|   | ca2949fbb4 | ||
|   | 17b373ac07 | ||
|   | 7aa66f438f | ||
|   | 73641fd78d | ||
|   | 64aac9d6ad | ||
|   | 2be13736c8 | ||
|   | ff4f5fec1d | ||
|   | 7d64f8abe4 | ||
|   | 88e6929e9f | ||
|   | 5fb0a995dd | ||
|   | 58a04ce1a5 | ||
|   | f74e0d9123 | ||
|   | c6249b82d4 | ||
|   | d6131c0b09 | ||
|   | e62c810b7c | ||
|   | 2851a1a7ef | ||
|   | 12cd2709d6 | ||
|   | bf54e58873 | ||
|   | 6b473e3a5c | ||
|   | 4b68abd963 | ||
|   | 0e764a2b3e | ||
|   | 9d1ed1eb0d | ||
|   | a09a3465a2 | ||
|   | 001969efaf | ||
|   | e7c515da9a | ||
|   | 8367c7dd49 | ||
|   | 55e6cae240 | ||
|   | 5553c3fb17 | ||
|   | 026265cb1e | ||
|   | 289c76a802 | ||
|   | 7c1bc1d6bc | ||
|   | 90cf0d32b5 | ||
|   | 88d934f922 | ||
|   | abf11bb03c | ||
|   | 2d1f50303d | ||
|   | 9fb7c4091f | ||
|   | 2fdec27ab0 | ||
|   | aff56469ed | ||
|   | 419cb7fbad | ||
|   | 7882851539 | ||
|   | 358299cf0e | ||
|   | 5a2af24869 | ||
|   | 6d35872af5 | ||
|   | 559dfdaa80 | ||
|   | 79c49bc926 | ||
|   | 76c538ad25 | ||
|   | e76e358d98 | ||
|   | 76f37671b4 | ||
|   | a9fc176c3c | ||
|   | 8b13e3c327 | ||
|   | c3cd6ad2d2 | ||
|   | 3444b9c9c8 | ||
|   | ca6fc9cd79 | ||
|   | 4747ae8b61 | ||
|   | b6c50d63a0 | ||
|   | 52ebf2055e | ||
|   | d0af2c2a98 | ||
|   | 10216af48a | ||
|   | d0c8d537f5 | ||
|   | 1903aaf351 | ||
|   | e6cdf1b995 | ||
|   | 2900d22cdc | ||
|   | dc072d4706 | ||
|   | 8ae14f146b | ||
|   | 57444c6c3f | ||
|   | 759719d124 | ||
|   | 59782973be | ||
|   | 6c647ea91c | ||
|   | c9763dabe1 | ||
|   | da82754659 | ||
|   | a3cc0ad18b | ||
|   | 2e8e5c2751 | ||
|   | a60d83b101 | ||
|   | 6d45265763 | ||
|   | 42e84b77e1 | ||
|   | 24121cfadb | ||
|   | 93093dd288 | ||
|   | 53381c04e6 | ||
|   | c6b64e57f1 | ||
|   | 9158426d0a | ||
|   | cef26853df | ||
|   | c0673884c5 | ||
|   | c98eb64598 | ||
|   | 8836bd4f3b | ||
|   | a58df29208 | ||
|   | 0d64a17d86 | ||
|   | c886c09cdb | ||
|   | e86d0007c6 | ||
|   | 624ee76e71 | ||
|   | 9406079cb7 | ||
|   | cd2de7f893 | ||
|   | 342061803e | ||
|   | 05b8111c19 | ||
|   | 747a0b1791 | ||
|   | c05586b53a | ||
|   | 0cfca4a618 | ||
|   | 2b8187f7ab | ||
|   | 291e7e7943 | ||
|   | 535d10f469 | ||
|   | 4cb58c0892 | ||
|   | 6721d27e3f | ||
|   | da9dd7c423 | ||
|   | 867eb41618 | ||
|   | 6fdff13480 | ||
|   | e581ead1ed | ||
|   | 7495206db2 | ||
|   | fe87d16d46 | ||
|   | 0db54386cd | ||
|   | 772258b0b8 | ||
|   | 3ef002e14d | ||
|   | b90d473ae5 | ||
|   | f5091d524b | ||
|   | ee5720df2c | ||
|   | afe2037985 | ||
|   | 878cd18144 | ||
|   | e8c8f67a09 | ||
|   | 7b29e7cf7e | ||
|   | 723d3e6871 | ||
|   | f063dee12a | ||
|   | db4b315d3d | ||
|   | 64d00b08a3 | ||
|   | 74736a941e | ||
|   | c40366f20a | ||
|   | d73e52ac2a | ||
|   | 9fac8a611f | ||
|   | 9c774a50f8 | ||
|   | 4f20ee1909 | ||
|   | 1440c5bc65 | ||
|   | 5829b2a7c1 | ||
|   | e3c4bc18d0 | ||
|   | 8555ec2f72 | ||
|   | 8d7cefbbbe | ||
|   | 5aa5896b22 | ||
|   | 8a55bdd89d | ||
|   | 6444ef6444 | ||
|   | 701ab6cc68 | ||
|   | 3c6ce7a943 | ||
|   | 1e54afd217 | ||
|   | 4e90fd3a06 | ||
|   | 0463c6bb0f | ||
|   | f31f986d66 | ||
|   | c203c8302b | ||
|   | dea4a7b389 | ||
|   | 0caf9b0706 | ||
|   | 5f655c9fd7 | ||
|   | 44aed53998 | ||
|   | f9ff621d4a | ||
|   | 23a5a75778 | ||
|   | a78475844a | ||
|   | 575cf2935e | ||
|   | ae43aa1bc0 | ||
|   | 0ba71c6a47 | ||
|   | 643f7d1803 | ||
|   | 5d06da4db2 | ||
|   | 2934b09361 | ||
|   | 97341d3fa0 | ||
|   | a95546ede0 | ||
|   | 7b44727b23 | ||
|   | debe648a98 | ||
|   | fda8cf77ed | ||
|   | 8aaab195c6 | ||
|   | 4096ddcbd0 | ||
|   | ae6cc11ad2 | ||
|   | dab7e527de | ||
|   | 8b92feac71 | ||
|   | 4beb3e5755 | ||
|   | 55f63229cd | ||
|   | 7827aeb695 | ||
|   | cce768aaac | ||
|   | 80b83c0624 | ||
|   | 73b683bb4d | ||
|   | d78a5c0863 | ||
|   | 683e5b6abe | ||
|   | 653b8f6352 | ||
|   | 9ec6afa375 | ||
|   | adff5382ca | ||
|   | 704aabd703 | ||
|   | f7b1ef0690 | ||
|   | 929982117f | ||
|   | 56a530d769 | ||
|   | 7ef75fb06b | ||
|   | 112a72abdf | ||
|   | 71813e03ee | ||
|   | 9b05b6ef28 | ||
|   | 54a87b25b3 | ||
|   | 55e97864bd | ||
|   | 95733c9490 | ||
|   | e19ae644f1 | ||
|   | ac4ea25267 | ||
|   | 611e4f34dc | ||
|   | faf017f333 | ||
|   | 5eec896615 | ||
|   | 17f35174ea | ||
|   | bf71b31123 | ||
|   | 9399a44c82 | ||
|   | c96418806f | ||
|   | 7945eddef6 | ||
|   | 0ede390fef | ||
|   | 85959a3b9b | ||
|   | 946c3a25b9 | ||
|   | f9d697128a | ||
|   | 532ef744f4 | ||
|   | 27a961814b | ||
|   | 5a5b65e9bf | ||
|   | 96673ad610 | ||
|   | a9025aea0d | ||
|   | f38ab0b973 | ||
|   | 150dac00cf | ||
|   | bcb1a9c5d3 | ||
|   | 2e55aea584 | ||
|   | 4f5a3f0df5 | ||
|   | e265b538cc | ||
|   | b186504718 | ||
|   | fc27890f13 | ||
|   | a583939767 | ||
|   | 52e3bcfd29 | ||
|   | 85d29a3f9d | ||
|   | 18944d389d | ||
|   | e90ac5d6a4 | ||
|   | cb9a6ae774 | ||
|   | 3ef09aa6b2 | ||
|   | 4db22e45a2 | ||
|   | f966d0b32c | ||
|   | ba3879a95a | ||
|   | c6cef0162d | ||
|   | 6d09aa86e9 | ||
|   | b711f0f9c6 | ||
|   | 8fefb3a4c9 | ||
|   | 400cdf0f26 | ||
|   | bce8c5a315 | ||
|   | f44dc2dd05 | ||
|   | df950d2fc5 | ||
|   | 5e1f804dd1 | ||
|   | 15de89f2f9 | ||
|   | df647a415c | ||
|   | fc66231f8e | ||
|   | 71df3e1566 | ||
|   | 168c22fc98 | ||
|   | 792ec23d7a | ||
|   | ff625253ce | ||
|   | 8c872c6b22 | ||
|   | 541f5bc0a6 | ||
|   | 77ff7b9df0 | ||
|   | 203cc5075e | ||
|   | 7abfcd06da | ||
|   | 724f81c7f3 | ||
|   | 4d2e98af7b | ||
|   | 08221fdda7 | ||
|   | c0f72970b9 | ||
|   | 18bc4a49e8 | ||
|   | f90b6dbed4 | ||
|   | d2d991ff34 | ||
|   | 190a5e175b | ||
|   | a05f5a91b8 | ||
|   | 4d02ff27be | ||
|   | df92b41d25 | ||
|   | 075af96338 | ||
|   | 64bbc55336 | ||
|   | 06c621acc1 | ||
|   | d040dc19bc | ||
|   | a38ae4a402 | ||
|   | 772063aade | ||
|   | 31c26354c5 | ||
|   | 64b331c5b2 | ||
|   | 94f8a145ec | ||
|   | b357afa30a | ||
|   | fee0437493 | ||
|   | 663b8864c1 | ||
|   | c6eafdde30 | ||
|   | f32ff95256 | ||
|   | 0452b75c3e | ||
|   | 97efd23ec8 | ||
|   | ea3e311528 | ||
|   | 0565454419 | ||
|   | 35c79c2f29 | ||
|   | e6089aec48 | ||
|   | dd3f7834c3 | ||
|   | f8cf18d880 | ||
|   | 2ac7e242b6 | ||
|   | d8b604a94f | ||
|   | a1f7b00981 | ||
|   | b7ee42198c | ||
|   | d66e4b7ff9 | ||
|   | 7f9789cf65 | ||
|   | b41c18e033 | ||
|   | 3c73a7ec6d | ||
|   | 182c09d952 | ||
|   | 9fb86aed04 | ||
|   | b8f7b13b05 | ||
|   | 1ee1d5b6d1 | ||
|   | 154c38c314 | ||
|   | 3c0626752a | ||
|   | 511ef77fb7 | ||
|   | 4df3ede500 | ||
|   | e83c080af8 | ||
|   | 054220db70 | ||
|   | 97bff010a8 | ||
|   | 487a3079cd | ||
|   | 2d5158c680 | ||
|   | 837be06b72 | ||
|   | 7e0d8dd4d7 | ||
|   | 146fb23d7d | ||
|   | 61455ffe29 | ||
|   | c267baafdc | ||
|   | 42248a306a | ||
|   | 694ec7ebe1 | ||
|   | e3b3f8fac1 | ||
|   | b390363b25 | ||
|   | f6f8cdbcf2 | ||
|   | f46f53b8a3 | ||
|   | a2fcae4383 | ||
|   | 483f043768 | ||
|   | f8e0f4f21f | ||
|   | 9c3613e96d | ||
|   | d9cacdc86d | ||
|   | aa3d2deeaa | ||
|   | e64912545a | ||
|   | b247be80cc | ||
|   | 343f2f1f33 | ||
|   | 2d590df900 | ||
|   | ba56b2b9fd | ||
|   | bf472b0c5e | ||
|   | f7961f34c5 | ||
|   | e369031a28 | ||
|   | 186d7bbfd9 | ||
|   | 60a11f8da5 | ||
|   | 1181fcdceb | ||
|   | 8cb9852058 | ||
|   | 53d46d1cbe | ||
|   | 275e1c8de9 | ||
|   | d46eca4c87 | ||
|   | 2927fb1597 | ||
|   | 8c72e011d2 | ||
|   | 69662e24c3 | ||
|   | 96099ffe98 | ||
|   | ae16b45c11 | ||
|   | cfee87d3ef | ||
|   | 5fcf5bc635 | ||
|   | 14bcb813cc | ||
|   | 6af19794b6 | ||
|   | 8316186695 | ||
|   | 85d3023cd5 | ||
|   | 78414dee29 | ||
|   | 084135141f | ||
|   | 467a21f028 | ||
|   | 6e284c44d6 | ||
|   | daf9a449e8 | ||
|   | 960268fd33 | ||
|   | 8c331da315 | ||
|   | b0d626d862 | ||
|   | a51fbd7316 | ||
|   | 063f372f3c | ||
|   | 987168b863 | ||
|   | 4ee40c3345 | ||
|   | 654daff7ce | ||
|   | 64e10e9619 | ||
|   | b89cffe98d | ||
|   | bd76ba702f | ||
|   | 5d52e9ce6b | ||
|   | 3c29027ca3 | ||
|   | 2ff3069d23 | ||
|   | 4198246351 | ||
|   | 2b6389b4dc | ||
|   | d7df75ae6c | ||
|   | 11c30eccb3 | ||
|   | ab8c6515b8 | ||
|   | d4ad36fa41 | ||
|   | 4d688be3df | ||
|   | d2b75f3501 | ||
|   | 46b78cb4ff | ||
|   | 5d6e0d0f37 | ||
|   | e19d0a37bb | ||
|   | dea3e2132e | ||
|   | 91c1ceefbd | ||
|   | 5c50989970 | ||
|   | 2a7e3b9c51 | ||
|   | ab302df0ae | ||
|   | 21e5809993 | ||
|   | c58afc67e8 | ||
|   | 8e344f2deb | ||
|   | c28f4ffb3f | ||
|   | 2a40240310 | ||
|   | 0e9fba9287 | ||
|   | 2e3dd2a30a | ||
|   | dda5f6559d | ||
|   | 4152e59638 | ||
|   | 9d5a92bce6 | ||
|   | 30172b92e6 | ||
|   | 8468a9d4c7 | ||
|   | 626cfb61ac | ||
|   | 9603f3fa4f | ||
|   | 619a1b9e53 | ||
|   | 62e76ad588 | ||
|   | 236d72685d | ||
|   | 72a5f7b1e2 | ||
|   | d1c16a90b4 | ||
|   | 33a9783ae5 | ||
|   | 4d64fd665e | ||
|   | 96443384fe | ||
|   | 69939f1edb | ||
|   | e3c0058942 | ||
|   | 3dc2361654 | ||
|   | ec2f709018 | ||
|   | ea06665c51 | ||
|   | 83b7010d6a | ||
|   | 71654cbe47 | ||
|   | e8f96e848a | ||
|   | 251abf21d4 | ||
|   | d103427932 | ||
|   | 74a4bd704c | ||
|   | c62f3e0e45 | ||
|   | a56a969433 | ||
|   | 1016f94bbb | ||
|   | e98cce9aee | ||
|   | d4bdb5d327 | ||
|   | db4378415e | ||
|   | 9b594880c8 | ||
|   | d44fc3db2f | ||
|   | 5133b0a0c0 | ||
|   | 815469304f | ||
|   | c651921c79 | ||
|   | d18c835221 | ||
|   | e38335077e | ||
|   | 34c150cf73 | ||
|   | 24cb3ba091 | ||
|   | c07eaef2d1 | ||
|   | 5df708ac9f | ||
|   | 38ffbe95e9 | ||
|   | 8bec4042fc | ||
|   | ee7ef89dfb | ||
|   | 54a5246061 | ||
|   | fa6eae2937 | ||
|   | 795be20ee1 | ||
|   | 87d3a06dcd | ||
|   | 8b5c917195 | ||
|   | c7da0e59ff | ||
|   | eb14acbe0c | ||
|   | 56860a6bef | ||
|   | 735687be21 | ||
|   | 592cdfa910 | ||
|   | f2ad1a0406 | ||
|   | fab0cc51b3 | ||
|   | 3a26acbdb2 | ||
|   | 8d0c31bcda | ||
|   | 0b99293d4c | ||
|   | 802153ff9b | ||
|   | ec73e2d237 | ||
|   | c32b020535 | ||
|   | c4441804e2 | ||
|   | 6c9d80d4e8 | ||
|   | a231f8d26c | ||
|   | 82af9320c0 | ||
|   | fceebf7388 | ||
|   | 08da258b63 | ||
|   | 5994926440 | ||
|   | 30f2da4215 | ||
|   | 1a2229f886 | ||
|   | 1e166490d9 | ||
|   | 1c57e6f80a | ||
|   | a6537a8748 | ||
|   | 842b75977b | ||
|   | 9b3dccf60c | ||
|   | 2d533e0cd8 | ||
|   | 9e1925c808 | ||
|   | 142d59be85 | ||
|   | d9a5e06b5b | ||
|   | b6889e9ac6 | ||
|   | dc82203e9b | ||
|   | 6a30d32e7d | ||
|   | d9780606b3 | ||
|   | fffa32df48 | ||
|   | cad49892d3 | ||
|   | f0a29721c9 | ||
|   | 13caf37991 | ||
|   | b71a602107 | ||
|   | e6ce0dd43a | ||
|   | 094a5214f1 | ||
|   | 0932fcd114 | ||
|   | f26643cea3 | ||
|   | 63b1689155 | ||
|   | 59dc929431 | ||
|   | deaadc33db | ||
|   | 0ba92a4f29 | ||
|   | 7dc5009ec7 | ||
|   | 0ee827afd3 | ||
|   | 551d1b7f86 | ||
|   | 7ed1b695f5 | ||
|   | 7ad31b9b33 | ||
|   | edd8992f7f | ||
|   | 96fe42cfcb | ||
|   | c3872b4a38 | ||
|   | 762945113d | ||
|   | 037e9230fc | ||
|   | 3f59ebf986 | ||
|   | e51e1d2b09 | ||
|   | 26cc49eb69 | ||
|   | 7987bb491c | ||
|   | 16b7ac5a87 | ||
|   | 5932cb8609 | ||
|   | dfe694d39f | ||
|   | 278624f2c8 | ||
|   | 899f42c070 | ||
|   | 8ce1d4d6a3 | ||
|   | 52225d703b | ||
|   | 81739af7cb | ||
|   | 25473222cc | ||
|   | 0b7be70935 | ||
|   | 818b71abd6 | ||
|   | 25575e8510 | ||
|   | 6c85adcf23 | ||
|   | 5dc92d7a40 | ||
|   | 4e2b966b80 | ||
|   | d34f8c3cb9 | ||
|   | 9049ecb1cf | ||
|   | 7bebea087c | ||
|   | 1c79e30436 | ||
|   | 1d7933349b | ||
|   | d002f67140 | ||
|   | da3447765b | ||
|   | cbf5663179 | ||
|   | b217fba235 | ||
|   | 7f7e6d5aba | ||
|   | 87c5a9d9a6 | ||
|   | 8ca1fe3f0a | ||
|   | 763ae8f1a6 | ||
|   | c65256d02b | ||
|   | bd2ac515d1 | ||
|   | 681f372889 | ||
|   | c2eec272e6 | ||
|   | bd720491a9 | ||
|   | a408226509 | ||
|   | c015e99e6e | ||
|   | de47a17be7 | ||
|   | d38fc490ad | ||
|   | 662167e792 | ||
|   | 36c91f03d9 | ||
|   | 33ccee26b5 | ||
|   | ed5cb991e3 | ||
|   | bea84ec2bf | ||
|   | 08c176e549 | ||
|   | 810ed50976 | ||
|   | 2684541693 | ||
|   | a5b12bac54 | ||
|   | fea1b06e43 | ||
|   | 182ca5d434 | ||
|   | facde9a75d | ||
|   | 41385640b9 | ||
|   | 7bad9db32e | ||
|   | af66f0a497 | ||
|   | 95e1b80f41 | ||
|   | 556e2eba95 | ||
|   | efe530cb17 | ||
|   | 34e7c99283 | ||
|   | 4157ea8bc3 | ||
|   | 550517bbf3 | ||
|   | eb910cd8a1 | ||
|   | 75131c4e8a | ||
|   | ee29ab95be | ||
|   | e97951fc51 | ||
|   | dfabdef60f | ||
|   | 5a87763193 | ||
|   | 6bb90f56fa | ||
|   | c883ae1350 | ||
|   | 09e25e6a02 | ||
|   | bf5d43054b | ||
|   | 63b3c65691 | ||
|   | f193da7f67 | ||
|   | 40f38c2c0a | ||
|   | db439ef804 | ||
|   | 56eb896a03 | ||
|   | 68d43e43b6 | ||
|   | c60517e49a | ||
|   | 3f59d261f2 | ||
|   | 4068d220e5 | ||
|   | 18968e7208 | ||
|   | 38656103c0 | ||
|   | 0f65b1bcc5 | ||
|   | a628821834 | ||
|   | 6ceff60c1e | ||
|   | d762a6ce58 | ||
|   | 75a8037a46 | ||
|   | 1179920790 | ||
|   | b323a160e3 | ||
|   | b157e9535e | ||
|   | 7668475bd6 | ||
|   | 8bda2a1fb7 | ||
|   | b092086b5b | ||
|   | 69a0d9034f | ||
|   | f5be8fd313 | ||
|   | 7f835d7f76 | ||
|   | ddbb7c5993 | ||
|   | 00a3fe39e8 | ||
|   | 7537fb88d4 | ||
|   | a81bc71a1e | ||
|   | 0a0aa0e2db | ||
|   | c56b94ae96 | ||
|   | e90712706d | ||
|   | eb0623331f | ||
|   | d15bd59109 | ||
|   | 60e0b19372 | ||
|   | 922eb937ff | ||
|   | 87573284f1 | ||
|   | a91c585f55 | ||
|   | 953ea21d5e | ||
|   | ecb00968bc | ||
|   | 50ad8adb2d | ||
|   | 16878caf09 | ||
|   | 5bc30c5493 | ||
|   | 85d89cf4c4 | ||
|   | db693f598b | ||
|   | 0494c770a1 | ||
|   | c473b62aed | ||
|   | f19ac5320e | ||
|   | 612e3aafbc | ||
|   | 0e97fec451 | ||
|   | e8c8626ee4 | ||
|   | d89e0f07f8 | ||
|   | e7f81a42ce | ||
|   | ac614148b8 | ||
|   | 5eb02b4901 | ||
|   | 65631525f6 | ||
|   | 969435cfe9 | ||
|   | c932f7a25b | ||
|   | 42d164dc57 | ||
|   | a7e60f80bd | ||
|   | 3dd5f313b7 | ||
|   | 883962c393 | ||
|   | 8a30ff1c76 | ||
|   | e47c354916 | ||
|   | 496f42805d | ||
|   | c3d34bda37 | ||
|   | bb6ede2b8f | ||
|   | 822400a1ba | ||
|   | e3e08843f1 | ||
|   | ce0d4f77fa | ||
|   | 94fdb4e974 | ||
|   | 4d425fc8a4 | ||
|   | c6cdfa2f5a | ||
|   | 0fff2e4f16 | ||
|   | 80a2172715 | ||
|   | 5a0a297634 | ||
|   | 948a133b7b | ||
|   | 2ee826c958 | ||
|   | 539409faf8 | ||
|   | 606e46e4d7 | ||
|   | a179cfd69a | ||
|   | d8379253d4 | ||
|   | c3344fbd68 | ||
|   | 4cebd6e84a | ||
|   | 90fbf9dbb0 | ||
|   | d365b9f634 | ||
|   | a2f06acaa4 | ||
|   | 8c90cbcbfb | ||
|   | a4a47772dc | ||
|   | 5dde1f4602 | ||
|   | 9dc0909eeb | ||
|   | 0ed2592e41 | ||
|   | 76cff98220 | ||
|   | 60604b6f51 | ||
|   | f410b7aecb | ||
|   | 1a61f2cee9 | ||
|   | 78a8293520 | ||
|   | 03cfb4fc8d | ||
|   | 144345a359 | ||
|   | fd2c01515e | ||
|   | 219570e08b | ||
|   | 69df556ff5 | ||
|   | 5f4a52574f | ||
|   | 5a1f6c5839 | ||
|   | 91d0342fe8 | ||
|   | 8cc236daf8 | ||
|   | d283ec69f7 | ||
|   | d1aea7596c | ||
|   | c934987b14 | ||
|   | 00c9f4a2e5 | ||
|   | 6605c1d07f | ||
|   | 7325d66c52 | ||
|   | a485061e22 | ||
|   | 1f63f50343 | ||
|   | cd3170dabd | ||
|   | 841cedc5f8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7f4882734d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e7d647d412 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 913d14a58a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 909272ec3d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7af40ffbbe | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9df79a3ec9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4f2eee06aa | ||
|   | 1b9cf76008 | ||
|   | d035a43ed6 | ||
|   | 95ee9a6e09 | ||
|   | 02a63cdcb3 | ||
|   | f02125dd47 | ||
|   | c11e813146 | ||
|   | a365849048 | ||
|   | a493c9f769 | ||
|   | a13f522b2a | ||
|   | 1ed70b2e2c | ||
|   | 86d5a599b7 | ||
|   | c226fc8d63 | ||
|   | bbf4e1c413 | ||
|   | a24a20a83d | ||
|   | 725600da8f | ||
|   | f74a32ed9b | ||
|   | e08e72dd10 | ||
|   | ce02e1e528 | ||
|   | 0b27d8a717 | ||
|   | 2782e7d26f | ||
|   | 2c83a05e80 | ||
|   | 467f68502a | ||
|   | d95b0dee6b | ||
|   | a1f3323fa5 | ||
|   | 494796a7f0 | ||
|   | 94f2c20d35 | ||
|   | c1deb9438d | ||
|   | ea86527c66 | ||
|   | d1a18fe266 | ||
|   | 737064da82 | ||
|   | 606cc85ff5 | ||
|   | dcfc8f1b30 | ||
|   | ebe4b84f14 | ||
|   | 699d4897db | ||
|   | fcdfd8d323 | ||
|   | db8625c31a | ||
|   | b65f265c55 | ||
|   | c55237d09c | ||
|   | ed698b7b82 | ||
|   | d4ff19f013 | ||
|   | 972fb8eb40 | ||
|   | 4de75448b6 | ||
|   | e8ef8f0004 | ||
|   | a319b30382 | ||
|   | 8278616eeb | ||
|   | 771f011506 | ||
|   | 826865869a | ||
|   | 3c77ae7b62 | ||
|   | 60c30ece10 | ||
|   | 76a0d0fee9 | ||
|   | d50624f0a0 | ||
|   | 4f9b015d1c | ||
|   | 4f10bd038d | ||
|   | 977af0a24d | ||
|   | f3ceb32a7c | ||
|   | 15da2de256 | ||
|   | 41ae0ccd6f | ||
|   | 344532662e | ||
|   | 99b365030e | ||
|   | 5cc940c08a | ||
|   | 1d02d9e0fe | ||
|   | 139523b763 | ||
|   | e7c83db9c7 | ||
|   | da47782b77 | ||
|   | 4685de88a7 | ||
|   | 0c28ac4907 | ||
|   | ae2d707c68 | ||
|   | 00a28193a0 | ||
|   | ea1818284b | ||
|   | d83efecc94 | ||
|   | 9a9fb37a78 | ||
|   | b6dc6c6984 | ||
|   | 517084b1fc | ||
|   | 27763e6898 | ||
|   | 57dde1da38 | ||
|   | 0bb961767c | ||
|   | 88515ce677 | ||
|   | 00562e840c | ||
|   | b7927ba386 | ||
|   | 9c363ff045 | ||
|   | 1dbce5e3e2 | ||
|   | 361a9ca1be | ||
|   | cde6514839 | ||
|   | 507e2f727e | ||
|   | 8028499d2b | ||
|   | c2c79c4a87 | ||
|   | d56f7f3390 | ||
|   | ef70d17194 | ||
|   | 9789b9a083 | ||
|   | e6311fdb13 | ||
|   | 2231c54dee | ||
|   | 20de9a5e35 | ||
|   | ec3a6d7097 | ||
|   | 9d99bf5af8 | ||
|   | 52911cc9fd | ||
|   | 6f71ba376d | ||
|   | 9f439aabba | ||
|   | 33ad60b1f3 | ||
|   | 010d3f8281 | ||
|   | e27c4bf1b9 | ||
|   | 11cfc58ffc | ||
|   | 91f38a8ddd | ||
|   | b56fed8ed5 | ||
|   | 4a93dadc1c | ||
|   | 3a5f55c471 | ||
|   | 2919e37586 | ||
|   | 077bdbfdef | ||
|   | 61ac024127 | ||
|   | 8db8d3f39e | ||
|   | e7032363d7 | ||
|   | 8a9dc26419 | ||
|   | 022f5a18c8 | ||
|   | eae4b2f2e9 | ||
|   | d285452dec | ||
|   | aced183a66 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 77b150c53b | ||
|   | 9b2ddfea5f | ||
|   | bf2bdaa6ff | ||
|   | 42525bb12a | ||
|   | bc243c1ea3 | ||
|   | a2517d1a1d | ||
|   | c0a60260c2 | ||
|   | 3654f247c4 | ||
|   | c4ddfef2ed | ||
|   | 28086111e1 | ||
|   | 60c9a6528f | ||
|   | 47d4dcfdf5 | ||
|   | eb8d1211ba | ||
|   | 495aad6a2d | ||
|   | 77676d18c8 | ||
|   | 68d90caab7 | ||
|   | 31fc43ed16 | ||
|   | 31802c9749 | ||
|   | 192aa89f95 | ||
|   | b0a0faff7e | ||
|   | b67f3438e9 | ||
|   | 02e4929a97 | ||
|   | fcc6a65e08 | ||
|   | f4ae939124 | ||
|   | 664acb2d0e | ||
|   | e6e7cecdb6 | ||
|   | 563d604812 | ||
|   | 012d744f4c | ||
|   | d4c7ca76ac | ||
|   | 1a6aae944f | ||
|   | 71e0241c94 | ||
|   | d838ef5b76 | ||
|   | d90a5c9154 | ||
|   | 9b79a411e0 | ||
|   | b8e0ec9edc | ||
|   | 57009057ae | ||
|   | 5db7b2e193 | ||
|   | d994c84901 | ||
|   | febfb97bb4 | ||
|   | a6c5e62923 | ||
|   | ac0390fec3 | ||
|   | 97b99867f2 | ||
|   | a55d5516a6 | ||
|   | b679163d01 | ||
|   | 76edcdbe45 | ||
|   | d8d51519c4 | ||
|   | 3446969121 | ||
|   | 0e0c35a701 | ||
|   | c9a6c9e20a | ||
|   | 3d2b982a94 | ||
|   | 6157d8331b | ||
|   | 3d0fc09fae | ||
|   | b975751710 | ||
|   | 4530d40537 | ||
|   | 94e3ac9b72 | ||
|   | e13fe97ebb | ||
|   | 4ad7632113 | ||
|   | 0709cac97f | ||
|   | 7dd4180fba | ||
|   | 558213490a | ||
|   | 78a69e3da8 | ||
|   | 140c78e5a7 | ||
|   | a8e18e0e22 | ||
|   | 2a8bb23625 | ||
|   | 936a4d1bc4 | ||
| ![imgbot[bot]](/assets/img/avatar_default.png)  | 69c34e8d00 | ||
|   | 1a2a190828 | ||
|   | 251cf1d76f | ||
|   | 52774bbe64 | ||
|   | 68a6758302 | ||
|   | 13e43a4f74 | ||
|   | b7d62d09ec | ||
|   | 321ec18173 | ||
|   | a44ac3306e | ||
|   | 951288ecf0 | ||
|   | cc8a7dd588 | ||
|   | 813c52f51e | ||
|   | be3298639d | ||
|   | e8d2959717 | ||
|   | e7680e08eb | ||
|   | bd792d7661 | ||
|   | 4920983f23 | ||
|   | 2756f553c6 | ||
|   | fc52e95ad0 | ||
|   | 5d1d6bc028 | ||
|   | b106acac91 | ||
|   | a5071db864 | ||
|   | bae874eb45 | ||
|   | 60da17940d | ||
|   | 385eeed732 | ||
|   | 2e908758d8 | ||
|   | a164f8ad95 | ||
|   | 372138dfea | ||
|   | 922a657c7e | ||
|   | 3409a51cca | ||
|   | 7174a55846 | ||
|   | 6656a59402 | ||
|   | 7612ead551 | ||
|   | fa78fe665d | ||
|   | c89aa7eb95 | ||
|   | 43f4c5b7cd | ||
|   | 2b6c566386 | ||
|   | 91ef328b6b | ||
|   | 5a9d9dc41d | ||
|   | a48e503caa | ||
|   | fe00cb9ad5 | ||
|   | ed0fdaddbd | ||
|   | 893795a31d | ||
|   | f1047f1ce5 | ||
|   | 9beddc941a | ||
|   | 3a6a01d2d6 | ||
|   | e0e4bdbdbc | ||
|   | b3daf79b6a | ||
|   | 81aa21915b | ||
|   | 5aadd80243 | ||
|   | 23350b1cbc | ||
|   | daff0977cb | ||
|   | c1e7d56ff8 | ||
|   | 61aef56f83 | ||
|   | 249f591403 | ||
|   | 36599c82d1 | ||
|   | 7615b1ea0a | ||
|   | d8de9f8eba | ||
|   | 8c0e65ce6f | ||
|   | 9aa24c0552 | ||
|   | 47bf06f432 | ||
|   | 99d291c71b | ||
|   | d51cafca74 | ||
|   | 7921f8cd43 | ||
| ![imgbot[bot]](/assets/img/avatar_default.png)  | 1b8467d5e5 | ||
|   | e13b2689b5 | ||
|   | 9c167acbd9 | ||
|   | 1bc531edbd | ||
|   | b4d0db9202 | ||
|   | bd2b681367 | ||
|   | f0edf81cc4 | ||
|   | e81ac05ba6 | ||
|   | 6279ce8f22 | ||
|   | 0fd20cf588 | ||
|   | f2d55e7f60 | ||
|   | 963b0db3d3 | ||
|   | aeca115a37 | ||
|   | 6f97142b59 | ||
|   | b31d1ce61d | ||
|   | b07cd37a16 | ||
|   | 69b74a46b9 | ||
|   | 882d829558 | ||
|   | 532821d503 | ||
|   | 522ce67498 | ||
|   | 0e046faf4a | ||
|   | d9092dc81f | ||
|   | 92a4e90026 | ||
|   | 07dccad5b1 | ||
|   | 146b0d2889 | ||
|   | 388565fb10 | ||
|   | da4ba51a74 | ||
|   | 1edcd136a4 | ||
|   | 9883c751da | ||
|   | f78b28b995 | ||
|   | 54d40420ad | ||
|   | ba1492f977 | ||
|   | efd0368e56 | ||
|   | a766a57af9 | ||
|   | 3bdd8a2d90 | ||
|   | 7ef1205f8b | ||
|   | e8db63e788 | ||
|   | 0bcef2453c | ||
|   | b9f549135c | ||
|   | 87b0017386 | ||
|   | cc8ff556d4 | ||
|   | 021f74da54 | ||
|   | f9389802d7 | ||
|   | 18dd172c97 | ||
|   | 1d5a54ff6f | ||
|   | 03e2c7eec6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0902727d1c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 496895634d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9414e9e258 | ||
|   | 357528d139 | ||
|   | c4efbdf4c7 | ||
|   | fb4a921cd9 | ||
|   | 683b242215 | ||
|   | a5660d6c82 | ||
|   | f632ec50c1 | ||
|   | a55d15214b | ||
|   | f1709a2cc2 | ||
|   | effa542958 | ||
|   | e8bf742c87 | ||
|   | 2e6652edce | ||
|   | 230c204b48 | ||
|   | 3755c600b1 | ||
|   | 24513fc0a3 | ||
|   | 0a79a6564a | ||
|   | 562bb5842b | ||
|   | ec3ca3032e | ||
|   | 890770c275 | ||
|   | 9ed58a1b4e | ||
|   | 08984be2fe | ||
|   | e3ade148ca | ||
|   | 34c0eff89f | ||
|   | 40aba47a47 | ||
|   | 6736f51134 | ||
|   | 9d826d6e52 | ||
|   | 902d9bc7a5 | ||
|   | b6c86e2845 | ||
|   | 34dffdfc8f | 
| @@ -4,31 +4,15 @@ executors: | ||||
|   default: | ||||
|     working_directory: /tmp/workspace | ||||
|     docker: | ||||
|       - image: misskey/ci:latest | ||||
|       - image: circleci/mongo:latest | ||||
|   with-redis: | ||||
|     working_directory: /tmp/workspace | ||||
|     docker: | ||||
|       - image: misskey/ci:latest | ||||
|       - image: circleci/mongo:latest | ||||
|       - image: misskey/ci:v11-node11 | ||||
|       - image: circleci/redis:latest | ||||
|       - image: circleci/postgres:latest | ||||
|   docker: | ||||
|     working_directory: /tmp/workspace | ||||
|     docker: | ||||
|       - image: docker:latest | ||||
|   alpine: | ||||
|     working_directory: /tmp/workspace | ||||
|     docker: | ||||
|       - image: alpine:latest | ||||
|  | ||||
| jobs: | ||||
|   ok: | ||||
|     executor: alpine | ||||
|     steps: | ||||
|       - run: | ||||
|           name: OK | ||||
|           command: | | ||||
|             echo -e '\033[0;32mOK\033[0;39m' | ||||
|   build: | ||||
|     executor: default | ||||
|     steps: | ||||
| @@ -64,8 +48,6 @@ jobs: | ||||
|           key: yarn-v1-arch-{{ arch }}-env-{{ .Environment.variableName }}-package-{{ checksum "package.json" }}-lock-{{ checksum "yarn.lock" }} | ||||
|           paths: | ||||
|             - node_modules | ||||
| #      - store_artifacts: | ||||
| #          path: built | ||||
|       - persist_to_workspace: | ||||
|           root: . | ||||
|           paths: | ||||
| @@ -75,22 +57,10 @@ jobs: | ||||
|       executor: | ||||
|         type: string | ||||
|         default: "default" | ||||
|       without_redis: | ||||
|         type: boolean | ||||
|         default: false | ||||
|     executor: <<parameters.executor>> | ||||
|     steps: | ||||
|       - attach_workspace: | ||||
|           at: /tmp/workspace | ||||
|       - when: | ||||
|           condition: <<parameters.without_redis>> | ||||
|           steps: | ||||
|             - run: | ||||
|                 name: Configure | ||||
|                 command: | | ||||
|                   mv .config/test.yml .config/test_redis.yml | ||||
|                   touch .config/test.yml | ||||
|                   cat .config/test_redis.yml | while IFS= read line; do if [[ "$line" = '# __REDIS__' ]]; then break; else echo "$line" >> .config/test.yml; fi; done | ||||
|       - run: | ||||
|           name: Test | ||||
|           command: | | ||||
| @@ -134,58 +104,55 @@ workflows: | ||||
|   version: 2 | ||||
|   nodejs: | ||||
|     jobs: | ||||
|       - ok: | ||||
|       - hold: | ||||
|           name: manual-build-trigger | ||||
|           type: approval | ||||
|           filters: | ||||
|             branches: | ||||
|               only: | ||||
|                 - l10n_develop | ||||
|                 - imgbot | ||||
|                 - patch-autogen | ||||
|               ignore: master | ||||
|       - build: | ||||
|           filters: | ||||
|             branches: | ||||
|               ignore: | ||||
|                 - l10n_develop | ||||
|                 - imgbot | ||||
|                 - patch-autogen | ||||
|       - test: | ||||
|           executor: with-redis | ||||
|           name: manual-build | ||||
|           requires: | ||||
|             - build | ||||
|             - manual-build-trigger | ||||
|           filters: | ||||
|             branches: | ||||
|               ignore: | ||||
| #                - master | ||||
|                 - l10n_develop | ||||
|                 - imgbot | ||||
|                 - patch-autogen | ||||
|               ignore: master | ||||
|       - build: | ||||
|           name: auto-build | ||||
|           filters: | ||||
|             branches: | ||||
|               only: master | ||||
|       - test: | ||||
|           without_redis: true | ||||
|           name: manual-test | ||||
|           requires: | ||||
|             - build | ||||
|             - manual-build | ||||
|           filters: | ||||
| #            branches: | ||||
| #              only: master | ||||
|             branches: | ||||
|               ignore: | ||||
| #                - master | ||||
|                 - l10n_develop | ||||
|                 - imgbot | ||||
|                 - patch-autogen | ||||
|               ignore: master | ||||
|       - test: | ||||
|           name: auto-test | ||||
|           requires: | ||||
|             - auto-build | ||||
|           filters: | ||||
|             branches: | ||||
|               only: master | ||||
|   docker: | ||||
|     jobs: | ||||
|       - hold: | ||||
|           name: manual-build-trigger | ||||
|           type: approval | ||||
|           filters: | ||||
|             branches: | ||||
|               ignore: master | ||||
|       - docker: | ||||
|           name: manual-build | ||||
|           requires: | ||||
|             - hold | ||||
|             - manual-build-trigger | ||||
|           filters: | ||||
|             branches: | ||||
|               ignore: master | ||||
|       - docker: | ||||
|           name: auto-build | ||||
|           with_deploy: true | ||||
|           filters: | ||||
|             branches: | ||||
|   | ||||
| @@ -6,8 +6,6 @@ mongodb: | ||||
|   db: misskey | ||||
|   user: syuilo | ||||
|   pass: '' | ||||
| drive: | ||||
|   storage: 'db' | ||||
| redis: | ||||
|   host: localhost | ||||
|   port: 6379 | ||||
|   | ||||
| @@ -6,8 +6,6 @@ mongodb: | ||||
|   db: test-misskey | ||||
|   user: admin | ||||
|   pass: '' | ||||
| drive: | ||||
|   storage: 'db' | ||||
| # __REDIS__ | ||||
| redis: | ||||
|   host: localhost | ||||
|   | ||||
							
								
								
									
										4
									
								
								.config/docker_example.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.config/docker_example.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| # db settings | ||||
| POSTGRES_PASSWORD=example-misskey-pass | ||||
| POSTGRES_USER=example-misskey-user | ||||
| POSTGRES_DB=misskey | ||||
| @@ -1,8 +1,16 @@ | ||||
| #━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ | ||||
| # Misskey configuration | ||||
| #━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ | ||||
|  | ||||
| #   ┌─────┐ | ||||
| #───┘ URL └───────────────────────────────────────────────────── | ||||
|  | ||||
| # Final accessible URL seen by a user. | ||||
| url: https://example.tld/ | ||||
|  | ||||
| #   ┌───────────────────────┐ | ||||
| #───┘ Port and TLS settings └─────────────────────────────────── | ||||
|  | ||||
| ### Port and TLS settings ###################################### | ||||
| # | ||||
| # Misskey supports two deployment options for public. | ||||
| # | ||||
| @@ -30,86 +38,76 @@ url: https://example.tld/ | ||||
| #   You need to set Certificate in 'https' section. | ||||
|  | ||||
| # To use option 1, uncomment below line. | ||||
| # port: 3000    # A port that your Misskey server should listen. | ||||
| #port: 3000    # A port that your Misskey server should listen. | ||||
|  | ||||
| # To use option 2, uncomment below lines. | ||||
| # port: 443 | ||||
| # | ||||
| # https: | ||||
| #   # path for certification | ||||
| #   key: /etc/letsencrypt/live/example.tld/privkey.pem | ||||
| #   cert: /etc/letsencrypt/live/example.tld/fullchain.pem | ||||
| #port: 443 | ||||
|  | ||||
| ################################################################ | ||||
| #https: | ||||
| #  # path for certification | ||||
| #  key: /etc/letsencrypt/live/example.tld/privkey.pem | ||||
| #  cert: /etc/letsencrypt/live/example.tld/fullchain.pem | ||||
|  | ||||
| #   ┌──────────────────────────┐ | ||||
| #───┘ PostgreSQL configuration └──────────────────────────────── | ||||
|  | ||||
| mongodb: | ||||
| db: | ||||
|   host: localhost | ||||
|   port: 27017 | ||||
|   port: 5432 | ||||
|  | ||||
|   # Database name | ||||
|   db: misskey | ||||
|  | ||||
|   # Auth | ||||
|   user: example-misskey-user | ||||
|   pass: example-misskey-pass | ||||
|  | ||||
| drive: | ||||
|   storage: 'db' | ||||
| #   ┌─────────────────────┐ | ||||
| #───┘ Redis configuration └───────────────────────────────────── | ||||
|  | ||||
|   # OR | ||||
| redis: | ||||
|   host: localhost | ||||
|   port: 6379 | ||||
|   #pass: example-pass | ||||
|  | ||||
|   # storage: 'minio' | ||||
|   # bucket: | ||||
|   # prefix: | ||||
|   # config: | ||||
|   #   endPoint: | ||||
|   #   port: | ||||
|   #   useSSL: | ||||
|   #   accessKey: | ||||
|   #   secretKey: | ||||
| #   ┌─────────────────────────────┐ | ||||
| #───┘ Elasticsearch configuration └───────────────────────────── | ||||
|  | ||||
|   # S3 example | ||||
|   # storage: 'minio' | ||||
|   # bucket: bucket-name | ||||
|   # prefix: files | ||||
|   # config: | ||||
|   #   endPoint: s3-us-west-2.amazonaws.com | ||||
|   #   region: us-west-2 | ||||
|   #   useSSL: true | ||||
|   #   accessKey: XXX | ||||
|   #   secretKey: YYY | ||||
|  | ||||
|   # S3 example (with CDN, custom domain) | ||||
|   # storage: 'minio' | ||||
|   # bucket: drive.example.com | ||||
|   # prefix: files | ||||
|   # baseUrl: https://drive.example.com | ||||
|   # config: | ||||
|   #   endPoint: s3-us-west-2.amazonaws.com | ||||
|   #   region: us-west-2 | ||||
|   #   useSSL: true | ||||
|   #   accessKey: XXX | ||||
|   #   secretKey: YYY | ||||
|  | ||||
| # If enabled: | ||||
| #  The first account created is automatically marked as Admin. | ||||
| autoAdmin: true | ||||
|  | ||||
| # | ||||
| # Below settings are optional | ||||
| # | ||||
|  | ||||
| # Redis | ||||
| #redis: | ||||
| #  host: localhost | ||||
| #  port: 6379 | ||||
| #  pass: example-pass | ||||
|  | ||||
| # Elasticsearch | ||||
| #elasticsearch: | ||||
| #  host: localhost | ||||
| #  port: 9200 | ||||
| #  pass: null | ||||
|  | ||||
| #   ┌───────────────┐ | ||||
| #───┘ ID generation └─────────────────────────────────────────── | ||||
|  | ||||
| # You can select the ID generation method. | ||||
| # You don't usually need to change this setting, but you can | ||||
| # change it according to your preferences. | ||||
|  | ||||
| # Available methods: | ||||
| # aid ... Short, Millisecond accuracy | ||||
| # meid ... Similar to ObjectID, Millisecond accuracy | ||||
| # ulid ... Millisecond accuracy | ||||
| # objectid ... This is left for backward compatibility | ||||
|  | ||||
| # ONCE YOU HAVE STARTED THE INSTANCE, DO NOT CHANGE THE | ||||
| # ID SETTINGS AFTER THAT! | ||||
|  | ||||
| id: 'aid' | ||||
|  | ||||
| #   ┌─────────────────────┐ | ||||
| #───┘ Other configuration └───────────────────────────────────── | ||||
|  | ||||
| # If enabled: | ||||
| #  The first account created is automatically marked as Admin. | ||||
| autoAdmin: true | ||||
|  | ||||
| # Whether disable HSTS | ||||
| #disableHsts: true | ||||
|  | ||||
| # Clustering | ||||
| #clusterLimit: 1 | ||||
|  | ||||
| # IP address family used for outgoing request (ipv4, ipv6 or dual) | ||||
| #outgoingAddressFamily: ipv4 | ||||
|   | ||||
| @@ -1,13 +0,0 @@ | ||||
| var user = { | ||||
| 	user: 'example-misskey-user', | ||||
| 	pwd: 'example-misskey-pass', | ||||
| 	roles: [ | ||||
| 	    { | ||||
| 		    role: 'readWrite', | ||||
| 		    db: 'misskey' | ||||
| 	    } | ||||
| 	] | ||||
| }; | ||||
|  | ||||
| db.createUser(user); | ||||
|  | ||||
							
								
								
									
										6
									
								
								.dockerignore
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						
									
										6
									
								
								.dockerignore
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							| @@ -5,8 +5,8 @@ | ||||
| .vscode | ||||
| Dockerfile | ||||
| build/ | ||||
| db/ | ||||
| docker-compose.yml | ||||
| node_modules/ | ||||
| mongo/ | ||||
| redis/ | ||||
| elasticsearch/ | ||||
| node_modules/ | ||||
| redis/ | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,7 @@ | ||||
| # PATH                                          OWNERS | ||||
| /.autogen/                                      @acid-chicken | ||||
| /.circleci/                                     @syuilo @acid-chicken | ||||
| /.config/                                       @syuilo @AyaMorisawa @mei23 @acid-chicken | ||||
| /.config/                                       @syuilo @AyaMorisawa @mei23 @acid-chicken @rinsuki | ||||
| # /.config/mongo_initdb_example.js              @khws4v1 | ||||
| /.github/                                       @syuilo @AyaMorisawa @acid-chicken | ||||
| /.vscode/                                       @acid-chicken | ||||
| @@ -12,7 +12,7 @@ | ||||
| # /docs/*.fr.md                                 @BoFFire | ||||
| # /docs/docker.*.md                             @khws4v1 | ||||
| /locales/                                       @syuilo | ||||
| /src/                                           @syuilo @AyaMorisawa @mei23 @acid-chicken | ||||
| /src/                                           @syuilo @AyaMorisawa @mei23 @acid-chicken @rinsuki | ||||
| # /src/crypto_key.cc                            @akihikodaki | ||||
| # /src/crypto_key.d.ts                          @akihikodaki | ||||
| /.dockerignore                                  @syuilo # @khws4v1 | ||||
|   | ||||
							
								
								
									
										14
									
								
								.github/ISSUE_TEMPLATE/01_bug-report.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/ISSUE_TEMPLATE/01_bug-report.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,30 +1,30 @@ | ||||
| --- | ||||
| name: Bug Report | ||||
| name: 🐛 Bug Report | ||||
| about: Create a report to help us improve | ||||
| title: '' | ||||
| labels: bug | ||||
| labels: ⚠️bug? | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| # Summary | ||||
| ## 💡 Summary | ||||
|  | ||||
| <!-- Tell us what the bug is --> | ||||
|  | ||||
| # Expected Behavior | ||||
| ## 🙂 Expected Behavior | ||||
|  | ||||
| <!--- Tell us what should happen --> | ||||
|  | ||||
| # Actual Behavior | ||||
| ## ☹️ Actual Behavior | ||||
|  | ||||
| <!--- Tell us what happens instead of the expected behavior --> | ||||
|  | ||||
| # Steps to Reproduce | ||||
| ## 📝 Steps to Reproduce | ||||
|  | ||||
| 1. | ||||
| 2. | ||||
| 3. | ||||
|  | ||||
| # Environment | ||||
| ## 📌 Environment | ||||
|  | ||||
| <!-- Tell us where on the platform it happens --> | ||||
|   | ||||
| @@ -1,31 +0,0 @@ | ||||
| --- | ||||
| name: Client-side Bug Report | ||||
| about: Create a report to help us improve | ||||
| title: '' | ||||
| labels: bug, client-side | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| # Summary | ||||
|  | ||||
| <!-- Tell us what the bug is --> | ||||
|  | ||||
| # Expected Behavior | ||||
|  | ||||
| <!--- Tell us what should happen --> | ||||
|  | ||||
| # Actual Behavior | ||||
|  | ||||
| <!--- Tell us what happens instead of the expected behavior --> | ||||
|  | ||||
| # Steps to Reproduce | ||||
|  | ||||
| 1. | ||||
| 2. | ||||
| 3. | ||||
|  | ||||
| # Environment | ||||
|  | ||||
| <!-- Tell us where on the platform it happens --> | ||||
| <!-- e.g. desktop or mobile version, your browser, your OS --> | ||||
| @@ -1,12 +1,12 @@ | ||||
| --- | ||||
| name: Feature Request | ||||
| name: ✨ Feature Request | ||||
| about: Suggest an idea for this project | ||||
| title: '' | ||||
| labels: feature | ||||
| labels: ✨Feature | ||||
| assignees: '' | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| # Summary | ||||
| ## Summary | ||||
| 
 | ||||
| <!-- Tell us what the suggestion is --> | ||||
| @@ -1,31 +0,0 @@ | ||||
| --- | ||||
| name: Server-side Bug Report | ||||
| about: Create a report to help us improve | ||||
| title: '' | ||||
| labels: bug, server-side | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| # Summary | ||||
|  | ||||
| <!-- Tell us what the bug is --> | ||||
|  | ||||
| # Expected Behavior | ||||
|  | ||||
| <!--- Tell us what should happen --> | ||||
|  | ||||
| # Actual Behavior | ||||
|  | ||||
| <!--- Tell us what happens instead of the expected behavior --> | ||||
|  | ||||
| # Steps to Reproduce | ||||
|  | ||||
| 1. | ||||
| 2. | ||||
| 3. | ||||
|  | ||||
| # Environment | ||||
|  | ||||
| <!-- Tell us where on the platform it happens --> | ||||
| <!-- e.g. your Node.js version, your OS --> | ||||
| @@ -1,12 +0,0 @@ | ||||
| --- | ||||
| name: Client-side Feature Request | ||||
| about: Suggest an idea for this project | ||||
| title: '' | ||||
| labels: client-side, feature | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| # Summary | ||||
|  | ||||
| <!-- Tell us what the suggestion is --> | ||||
| @@ -1,12 +0,0 @@ | ||||
| --- | ||||
| name: Server-side Feature Request | ||||
| about: Suggest an idea for this project | ||||
| title: '' | ||||
| labels: feature, server-side | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| # Summary | ||||
|  | ||||
| <!-- Tell us what the suggestion is --> | ||||
							
								
								
									
										2
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| # Summary | ||||
| ## Summary | ||||
|  | ||||
| <!-- | ||||
|   - | ||||
|   | ||||
							
								
								
									
										25
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										25
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,21 +1,34 @@ | ||||
| # Visual Studio Code | ||||
| /.vscode | ||||
|  | ||||
| # Intelij-IDEA | ||||
| /.idea | ||||
|  | ||||
| # Node.js | ||||
| /node_modules | ||||
|  | ||||
| # yarn | ||||
| yarn.lock | ||||
|  | ||||
| # config | ||||
| /.config/* | ||||
| !/.config/example.yml | ||||
| !/.config/mongo_initdb_example.js | ||||
| /.vscode | ||||
| /node_modules | ||||
|  | ||||
| # misskey | ||||
| /build | ||||
| /built | ||||
| built | ||||
| /data | ||||
| /.cache-loader | ||||
| /db | ||||
| /elasticsearch | ||||
| npm-debug.log | ||||
| *.pem | ||||
| run.bat | ||||
| api-docs.json | ||||
| *.log | ||||
| /redis | ||||
| /mongo | ||||
| /elasticsearch | ||||
| *.code-workspace | ||||
| yarn.lock | ||||
| .DS_Store | ||||
| /files | ||||
| ormconfig.json | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| v11.7.0 | ||||
| v12.1.0 | ||||
|   | ||||
							
								
								
									
										740
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										740
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,6 +1,746 @@ | ||||
| ChangeLog | ||||
| ========= | ||||
|  | ||||
| If you encounter any problems with updating, please try the following: | ||||
| 1. `npm run clean` or `npm run cleanall` | ||||
| 2. Retry update (Don't forget `npm i`) | ||||
|  | ||||
| Migration | ||||
| ------------------------------ | ||||
| #### 1 | ||||
| ``` | ||||
| npm i -g ts-node | ||||
| ``` | ||||
|  | ||||
| #### 2 | ||||
| ``` | ||||
| npm run migrate | ||||
| ``` | ||||
|  | ||||
| How to migrate to v11 from v10 | ||||
| ------------------------------ | ||||
| ### 移行の注意点 | ||||
| **以下のデータは引き継がれません** | ||||
| * 通知 | ||||
| * リモートの投稿 | ||||
| * リバーシの対局 | ||||
|  | ||||
| ### 手順 | ||||
| 1. v11をインストールしたい場所に syuilo/misskey をクローン | ||||
| 2. config を設定する | ||||
| 	* PostgreSQL(`db`)の設定とは別に、v10からMongoDBの設定をコピペしてくる(例は下にあります) | ||||
| 	* `id`の設定を`meid`または`objectid`にする | ||||
|  | ||||
| ``` yml | ||||
| db: | ||||
|   host: localhost | ||||
|   port: 5432 | ||||
|   db: misskey | ||||
|   user: x | ||||
|   pass: x | ||||
|  | ||||
| mongodb: | ||||
|   user: x | ||||
|   pass: x | ||||
|   host: localhost | ||||
|   port: 27017 | ||||
|   db: misskey | ||||
| ``` | ||||
| 3. migration ブランチに切り替え | ||||
| 4. `npm i` | ||||
| 5. `npm run build` | ||||
| 6. `npm run init` | ||||
| 7. `npm run migrate` | ||||
| 8. master ブランチに戻す | ||||
| 9. enjoy | ||||
|  | ||||
| 11.18.0 (2019/05/21) | ||||
| -------------------- | ||||
| ### ✨Improvements | ||||
| * デザインの調整 | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * 投稿の削除が連合しない問題を修正 | ||||
|  | ||||
| 11.17.1 (2019/05/20) | ||||
| -------------------- | ||||
| ### 🐛Fixes | ||||
| * トークの通知インジケーターが点灯し続ける問題を修正 | ||||
| * ユーザーグループの読み込みでエラーになることがある問題を修正 | ||||
|  | ||||
| 11.17.0 (2019/05/19) | ||||
| -------------------- | ||||
| ### 注意 | ||||
| このアップデートを適用した後、プロセスを起動(もしくは再起動)する前に[マイグレーション](#migration)の手順を実行してください | ||||
|  | ||||
| ### ✨Improvements | ||||
| * ユーザーグループを招待制に | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * フォロー申請が表示されない問題を修正 | ||||
| * トークの通知インジケーターが点灯し続ける問題を修正 | ||||
| * 自分を自分のグループから削除できる問題を修正 | ||||
| * 新しいタブでグループトークを開くことができない問題を修正 | ||||
|  | ||||
| 11.16.1 (2019/05/19) | ||||
| -------------------- | ||||
| ### 🐛Fixes | ||||
| * Dockerで起動できない問題を修正 | ||||
|  | ||||
| 11.16.0 (2019/05/19) | ||||
| -------------------- | ||||
| ### 注意 | ||||
| このアップデートを適用した後、プロセスを起動(もしくは再起動)する前に[マイグレーション](#migration)の手順を実行してください | ||||
|  | ||||
| ### ✨Improvements | ||||
| * ユーザーグループ機能を追加 | ||||
| * ページに「いいね」できるように | ||||
| * UIの改善 | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * トークを読み込むときに最大数指定できなかった問題を修正 | ||||
|  | ||||
| 11.15.0 (2019/05/16) | ||||
| -------------------- | ||||
| ### ✨Improvements | ||||
| * 管理画面でreCAPTCHAのプレビューを表示するように | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * オブジェクトストレージのリージョンの設定が反映されない問題を修正 | ||||
|  | ||||
| 11.14.0 (2019/05/16) | ||||
| -------------------- | ||||
| ### 注意 | ||||
| このバージョンからオブジェクトストレージの設定は設定ファイルではなく管理画面から行うようになりました。 | ||||
| オブジェクトストレージを使用している場合、アップデートした後管理画面にアクセスしオブジェクトストレージの設定を再度行ってください。 | ||||
|  | ||||
| ### ✨Improvements | ||||
| * 特定のユーザーのファイルをすべて削除できるように | ||||
| * インスタンスの設定画面を整理 | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * GIF画像のサムネイルが生成されないのを修正 | ||||
| * 管理画面の「ログ」で複数の除外条件を設定できない問題を修正 | ||||
|  | ||||
| 11.13.0 (2019/05/14) | ||||
| -------------------- | ||||
| ### 注意 | ||||
| このアップデートを適用した後、プロセスを起動(もしくは再起動)する前に[マイグレーション](#migration)の手順を実行してください | ||||
|  | ||||
| ### ✨Improvements | ||||
| * 利用規約URL、リポジトリURL、フィードバックURLを設定できるように | ||||
| * 特定のインスタンスのファイルをすべて削除できるように | ||||
| * _blankで外部リンクされる可能性がある箇所にnoopenerを追加 | ||||
| * ユーザーや外部インスタンスが生成するリンクにnofollowを追加 | ||||
| * リモートのユーザーページやノートページにnoindexを追加 | ||||
| * 自分のユーザーメニューにはミュートなどを表示しないように | ||||
| * デザインの調整 | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * インスタンスブロックを設定できない問題を修正 | ||||
| * ピン留め投稿の表示順がおかしい問題を修正 | ||||
| * 設定の「アップデートを確認」でメッセージが正しく表示されない問題を修正 | ||||
| * Firefoxで自分のメニューが開けない問題を修正 | ||||
| * Welcomeページのタグクラウドが動かない問題を修正 | ||||
|  | ||||
| 11.12.0 (2019/05/10) | ||||
| -------------------- | ||||
| ### 注意 | ||||
| このアップデートを適用した後、プロセスを起動(もしくは再起動)する前に[マイグレーション](#migration)の手順を実行してください | ||||
|  | ||||
| ### ✨ Improvements | ||||
| * インスタンス運営者がおすすめアカウントを設定できるように | ||||
| * MisskeyPagesでNAME環境変数がNULLにならないように | ||||
| * MisskeyPagesにNULL環境変数を追加 | ||||
| * MisskeyPagesで変数を並べ替えられるように | ||||
| * MisskeyPagesのテキストのリスト内で変数埋め込みできるように | ||||
| * 自分の指定した投稿のRenoteを全て解除するAPIを追加 | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * Noteをpull取得した時にhost名がvalidateされていない問題を修正 | ||||
| * みつけるで人気のタグが表示されない問題を修正 | ||||
|  | ||||
| ### その他 | ||||
| * アカウントのisVerifiedフラグを廃止 | ||||
|  | ||||
| 11.11.2 (2019/05/07) | ||||
| -------------------- | ||||
| ### 🐛Fixes | ||||
| * IPv4 onlyホストからDualstackホストにAP deliverできない問題を修正 | ||||
| * ストリーミングに接続するまでラグがある問題を修正 | ||||
| * 2段階認証のコードが0から始まる時正しく入力できない問題を修正 | ||||
| * ユーザーの更新日時が新しい順で更新日時がnullのユーザーが先頭に来る問題を修正 | ||||
| * 値選択時の問題を修正 | ||||
| * リバーシでマップの変更が反映されない問題を修正 | ||||
| * リバーシで対局終了直後に盤面を巻き戻してもすぐ最終ターンまでリセットされる問題を修正 | ||||
|  | ||||
| 11.11.1 (2019/05/05) | ||||
| -------------------- | ||||
| ### 🐛Fixes | ||||
| * MisskeyPagesのリストから選択関数が使えない問題を修正 | ||||
|  | ||||
| 11.11.0 (2019/05/05) | ||||
| -------------------- | ||||
| ### ✨ Improvements | ||||
| * MisskeyPagesにリストから選択関数を追加 | ||||
| * MisskeyPagesに確率を指定できるテキストランダム選択関数を追加 | ||||
| * 外部サービス連携ログインリンクにアイコン追加 | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * MisskeyPagesでifを入れ子にできなくなっていた問題を修正 | ||||
| * MisskeyPagesで数値入力を作成するとテキスト入力になる問題を修正 | ||||
| * 外部サービス連携に関する問題を修正 | ||||
|  | ||||
| 11.10.1 (2019/05/04) | ||||
| -------------------- | ||||
| ### 🐛Fixes | ||||
| * MisskeyPagesでページブロックを削除できなくなっていた問題を修正 | ||||
|  | ||||
| ### その他 | ||||
| * Node.js v12対応 | ||||
|  | ||||
| 11.10.0 (2019/05/03) | ||||
| -------------------- | ||||
| ### 注意 | ||||
| このアップデートを適用した後、プロセスを起動(もしくは再起動)する前に[マイグレーション](#migration)の手順を実行してください | ||||
|  | ||||
| ### ✨ Improvements | ||||
| * MisskeyPagesに割った余りを求める関数を追加 | ||||
| * Mastodon v2.8.0 のフォローリストをインポートできるように | ||||
| * エクスポートリクエストに失敗したらエラーを表示するように | ||||
| * エクスポートファイルでは同一ハッシュチェックをしないように | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * 2段階認証を設定するとログインできなくなる問題を修正 | ||||
| * ファイルをアップロードできないことがある問題を修正 | ||||
| * リモートファイルをキャッシュしない設定だとサムネイル時にオリジナル画像が表示されない問題を修正 | ||||
| * 外部サービス連携の不具合を修正 | ||||
|  | ||||
| 11.9.0 (2019/05/02) | ||||
| ------------------- | ||||
| ### ✨ Improvements | ||||
| * MisskeyPagesで編集時にページブロックをドラッグで並べ替えられるように | ||||
| * MisskeyPagesにカウンターボタンブロックを追加 | ||||
|  | ||||
| 11.8.1 (2019/05/02) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * リモートファイルをキャッシュしないオプション有効時にファイルが作成できない問題を修正 | ||||
|  | ||||
| 11.8.0-2 (2019/05/01) | ||||
| ------------------- | ||||
| * 11.8.0 のリリース内容が 11.7.0 と同一だったのを修正 | ||||
|  | ||||
| 11.8.0 (2019/05/01) | ||||
| ------------------- | ||||
| ### ✨ Improvements | ||||
| * MisskeyPagesで関数を作成できるように | ||||
| * MisskeyPagesでソースを表示できるように | ||||
| * MisskeyPagesにシードを与えるランダム関数を追加 | ||||
| * MisskeyPagesに複数行テキストをテキストのリストに変換する関数を追加 | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * APIドキュメントが見れなくなっていたのを修正 | ||||
| * mention (あなた宛て) streaming にミュートが効かない問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 11.7.0 (2019/04/30) | ||||
| ------------------- | ||||
| ### ✨ Improvements | ||||
| * MisskeyPagesに ifブロック を追加 | ||||
| * MisskeyPagesに テキストエリア を追加 | ||||
| * MisskeyPagesに 複数行テキスト入力 を追加 | ||||
| * MisskeyPagesに 投稿フォーム を追加 | ||||
| * MisskeyPagesに 変換系関数 を追加 | ||||
| * MisskeyPagesに 環境変数 URL を追加 | ||||
| * MisskeyPagesでボタンやスイッチなどのテキストに変数使えるように | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * OGPのサイト名を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 11.6.0 (2019/04/29) | ||||
| ------------------- | ||||
| ### ✨ Improvements | ||||
| * AiScriptにいくつかの文字列操作関数を追加 | ||||
| * ページ編集画面にページへのリンクを表示するように | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * MisskeyPagesで数値入力が文字列として扱われる問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 11.5.1 (2019/04/29) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * MisskeyPagesで環境変数を別の変数内で使えない問題を修正 | ||||
| * MisskeyPagesで値が0の変数が表示されない問題を修正 | ||||
|  | ||||
| 11.5.0 (2019/04/29) | ||||
| ------------------- | ||||
| ### 注意 | ||||
| このアップデートを適用した後、プロセスを起動(もしくは再起動)する前に[マイグレーション](migration)の手順を実行してください | ||||
|  | ||||
| ### New features | ||||
| #### MisskeyPages | ||||
| ページ(記事)を作成できるように。 | ||||
|  | ||||
| * 後から何度でも編集できる | ||||
| * アイキャッチを設定できる | ||||
| * フォントを設定できる | ||||
| * 画像を好きな位置に挿入できる | ||||
| * URLを決められる | ||||
| * タイトルを設定できる | ||||
| * 見出しを設定できる | ||||
| * ページの要約を設定できる(URLプレビュー時などに便利) | ||||
| * 変数や式(aka AiScript)を使用して動的なページも作れる | ||||
| * 目次自動生成(coming soon) | ||||
|  | ||||
| ページを気に入ったら「いいね」しよう (coming soon) | ||||
|  | ||||
| ### ✨ Improvements | ||||
| * APIコンソールでパラメータテンプレートを表示するように | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * おすすめユーザーに自分自身が含まれる問題を修正 | ||||
| * ユーザーサジェストで表示名が変わらない問題を修正 | ||||
|  | ||||
| 11.4.0 (2019/04/25) | ||||
| ------------------- | ||||
| ### ✨ Improvements | ||||
| * 検索でローカルの投稿のみに絞れるように | ||||
| * 検索で特定のインスタンスの投稿のみに絞れるように | ||||
| * 検索で特定のユーザーの投稿のみに絞れるように | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * 投稿が増殖する問題を修正 | ||||
| * ストリームで過去の投稿が流れてくる問題を修正 | ||||
| * モバイル版のユーザーページで遷移してもユーザー名が変わらない問題を修正 | ||||
| * お知らせを切り替えても内容が変わらない問題を修正 | ||||
|  | ||||
| 11.3.1 (2019/04/24) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * Webからファイルがアップロードできない問題を修正 | ||||
|  | ||||
| 11.3.0 (2019/04/24) | ||||
| ------------------- | ||||
| ### ✨ Improvements | ||||
| * お知らせにMFMを使えるように | ||||
| * お知らせに画像を添付できるように | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * 投稿のタグ検索APIで大文字小文字が区別されていたのを修正 | ||||
| * 公開範囲がホームの投稿がグローバルTLに流れる問題を修正 | ||||
| * モバイルビューの投稿詳細にて acct が長いとアイコンが圧迫面接される問題を修正 | ||||
|  | ||||
| 11.2.2 (2019/04/22) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * 2段階認証を有効にするとログインできない問題を修正 | ||||
| * リモートユーザーの修復処理が自動的に実行されない問題を修正 | ||||
| * リモートユーザー情報が更新されない問題を修正 | ||||
|  | ||||
| 11.2.1 (2019/04/21) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * MEIDが25桁になっているのを修正 | ||||
| * リモートユーザー情報が更新されない問題を修正 | ||||
|  | ||||
| 11.2.0 (2019/04/18) | ||||
| ------------------- | ||||
| ### ✨ Improvements | ||||
| * 検索で日付(日時)を入力するとタイムラインをその時点まで遡るように | ||||
| * APIコンソールでエンドポイントをサジェストするように | ||||
| * モバイル版でドライブのメニューを使いやすく | ||||
| * サイレンス時に確認を表示するように | ||||
| * ユーザーメニューでブロックなどの操作を行う時に確認するように | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * アプリケーション連携画面でパーミッションが表示されない問題を修正 | ||||
| * アンケートウィジットでもMFMを使用するように | ||||
| * フォローしてないユーザーのホーム投稿がSTLに流れてくる問題を修正 | ||||
| * モバイル版でウィジェットを設定できない問題を修正 | ||||
| * スプラッシュがクリックに反応するように | ||||
|  | ||||
| 11.1.6 (2019/04/18) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * 未認知ユーザーからActivityが飛んできた場合に処理できない問題を修正 | ||||
| * その投稿を見たのにも関わらずメンションインジケーターが点灯し続ける問題を修正 | ||||
| * ハッシュタグの判定を改善 | ||||
| * サーバーのエラーハンドリングを改善 | ||||
|  | ||||
| 11.1.5 (2019/04/17) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * ユーザー名に含まれているカスタム絵文字が表示されないことがある問題を修正 | ||||
| * 壁紙の設定ができない問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 11.1.4 (2019/04/17) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * タイムライン取得時に削除されたファイルを添付している投稿が含まれているとサーバーでエラーになる問題を修正 | ||||
| * 管理画面のインスタンスメニューで変更前の設定が読み込まれないことがある問題を修正 | ||||
| * 猫ではないのに猫のままで表示される問題を修正 | ||||
| * admin/driveのアイコンがずれてる問題を修正 | ||||
| * チャートで大きな数値を扱えない問題を修正 | ||||
| * UIの修正 | ||||
|  | ||||
| 11.1.3 (2019/04/16) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * アプリからAPIにリクエストするときにランダムなユーザーがリクエストしたことになる問題を修正 | ||||
|  | ||||
| 11.1.2 (2019/04/15) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * 画像描画の依存関係を変更 | ||||
| * リモートユーザーのファイルを削除するときに古い方からではなく新しい方から削除されるのを修正 | ||||
| * リアクションしてないのにリアクションしたことになる問題を修正 | ||||
| * APIドキュメントの修正 | ||||
|  | ||||
| 11.1.1 (2019/04/15) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * Metaタグの application-name を Misskey で固定するように修正 | ||||
| * トークメッセージが既読にならない問題を修正 | ||||
| * デフォルトでHTLを表示するように | ||||
|  | ||||
| 11.1.0 (2019/04/15) | ||||
| ------------------- | ||||
| ### ✨ Improvements | ||||
| * アイコン未設定時にランダムな画像を表示するように | ||||
| * 管理者やモデレーターはレートリミット無効に | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * メンションの「あなた」インジケーターが表示されない問題を修正 | ||||
| * ブロックAPIでエラーが発生する問題を修正 | ||||
| * プッシュ通知の購読に失敗する問題を修正 | ||||
|  | ||||
| 11.0.3 (2019/04/15) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * ハッシュタグ検索APIが動作しない問題を修正 | ||||
| * モデレーターなのにアカウントメニューに「管理」が表示されない問題を修正 | ||||
| * プッシュ通知の購読に失敗する問題を修正 | ||||
| * ユーザー取得APIでユーザーを指定しない場合エラーになる問題を修正 | ||||
|  | ||||
| 11.0.2 (2019/04/15) | ||||
| ------------------- | ||||
| ### 🐛Fixes | ||||
| * アプリが作成できない問題を修正 | ||||
| * 「ハイライト」が表示されない問題を修正 | ||||
| * リモートの投稿に添付されている画像が小さい問題を修正 | ||||
| * モバイル版でリストの名前が表示されない問題を修正 | ||||
| * APIドキュメントにパーミッション一覧を追加 | ||||
|  | ||||
| 11.0.1 (2019/04/15) | ||||
| ------------------- | ||||
| ### ✨ Improvements | ||||
| * 不要な依存関係を削除 | ||||
|  | ||||
| 11.0.0 daybreak (2019/04/14) | ||||
| ---------------------------- | ||||
| ### ✨ Improvements | ||||
| * **データベースがMongoDBからPostgreSQLに変更されました** | ||||
| * **Redisが必須に** | ||||
| * アカウントを完全に削除できるように | ||||
| * 投稿フォームで添付ファイルの閲覧注意を確認/設定できるように | ||||
| * ミュート/ブロック時にそのユーザーの投稿のウォッチをすべて解除するように | ||||
|  | ||||
| ### 🐛Fixes | ||||
| * フォロー申請数が実際より1すくなくなる問題を修正 | ||||
| * リストからアカウント削除したユーザーを削除できない問題を修正 | ||||
| * リストTLでフォローしていないユーザーの非公開投稿が流れる問題を修正 | ||||
| * リストTLでダイレクト投稿が流れない問題を修正 | ||||
| * ミュートしているユーザーの投稿がタイムラインに流れてくることがある問題を修正 | ||||
|  | ||||
| ### APIの破壊的変更 | ||||
| * v10時点で deprecated だったパラメータなどを削除 | ||||
| * ユーザーリストの title が name に | ||||
| * リバーシの対局の`settings`プロパティがなくなり、その中にあったプロパティがすべて上の階層に | ||||
|   * 例えば`game.settings.map`は`game.map`になる | ||||
|  | ||||
| ### 既知の問題 | ||||
| * ユーザー認証無しでのアプリが作成できない | ||||
|   * 依存ライブラリの問題と思わるため、対応が難しい | ||||
|  | ||||
| ### Migration | ||||
| coming soon... | ||||
|  | ||||
| 10.100.0 | ||||
| ---------- | ||||
| * ユーザーリストでフォローボタンを表示するように | ||||
| * ドライブのファイルのサムネイルを修正 | ||||
| * 投稿ウィジットでローカルのみの公開範囲で投稿できない問題を修正 | ||||
| * TLを遡った時に抜けがある時がある問題を修正 | ||||
| * ユーザータイムラインが投稿日時順ではなくなっているのを修正 | ||||
| * 10.99.0 でチャートのレンダリングがおかしい問題を修正 | ||||
|  | ||||
| 10.99.0 | ||||
| ---------- | ||||
| * manifest.json にインスタンス名を反映させるように | ||||
| * Metaに投稿やユーザーのIDを設定するように | ||||
| * 設定でポートが指定されていない場合、環境変数を参照するように | ||||
| * フォローインポートで途中にエラーになるユーザーがいると途中で終了してしまう問題を修正 | ||||
| * フォローインポートで自分が含まれていた場合自分をフォローしてしまう問題を修正 | ||||
| * ServiceWorkerの設定がUIで有効にならない問題を修正 | ||||
| * ユーザー一覧でのユーザーの自己紹介が複数行になることがある問題を修正 | ||||
| * フォローインポートでAPI limitに達していても正常にリクエストされたように表示されてしまう問題を修正 | ||||
| * DBに保存されたrepository urlを変更する方法がない問題を修正 | ||||
| * デスクトップDeckだとviaが投稿内に2箇所表示される問題を修正 | ||||
| * デザインの調整 | ||||
| * 依存関係の更新 | ||||
| * ローカリゼーション | ||||
|  | ||||
| 10.98.3 | ||||
| ---------- | ||||
| * リアクションのカスタム絵文字の情報がNoteに添付されない問題を修正 | ||||
| * フォルダーの移動をするとき親フォルダーに自分自身を指定できてしまう問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.98.2 | ||||
| ---------- | ||||
| * 他のインスタンスから添付画像が見れない問題を修正 | ||||
|  | ||||
| 10.98.1 | ||||
| ---------- | ||||
| * ドライブのファイルのサムネイルが表示されない問題を修正 | ||||
| * APでカスタム絵文字を送る時に常にimage/pngで送っている問題を修正 | ||||
| * いくらいじってもページリロードするとmisskeyのテーマがdark(future)になっちゃう問題を修正 | ||||
|  | ||||
| 10.98.0 | ||||
| ---------- | ||||
| * ドライブのファイルダウンロード時に元のファイル名を尊重するように | ||||
| * ドライブで画像以外のファイルを分かりやすく表示するように | ||||
| * TwemojiのCDNを変更 | ||||
| * モバイルで通知の設定がない問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.97.2 | ||||
| ---------- | ||||
| * ビルド時に警告が出ないように修正 | ||||
|  | ||||
| 10.97.1 | ||||
| ---------- | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.97.0 | ||||
| ---------- | ||||
| * リアクションに絵文字やカスタム絵文字を使えるように | ||||
| * 不明なリアクションのフォールバックに star を使えるように | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.96.0 | ||||
| ---------- | ||||
| * 連合ユーザーの投稿に対してActivityPubオブジェクトを要求されたら元のインスタンスにリダイレクトするように | ||||
| * updatePersonを試行した時点でもlastFetchedAtを更新するように | ||||
| * 管理画面でリモートインスタンスの登録日時を表示 | ||||
| * ユーザーサジェストが機能しなくなっていた問題を修正 | ||||
| * 最近使ったハッシュタグ表示が機能していない問題を修正 | ||||
| * バグ修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.95.0 | ||||
| ---------- | ||||
| * ジョブを一覧できるように | ||||
| * MFMでURLを明示する構文の追加 | ||||
| * Articleタイプのアクティビティを受け入れるように | ||||
| * 凍結されたユーザーをサジェストしないように | ||||
| * ファビコンが保存されないのを修正 | ||||
| * キューのジョブクリアの動作を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.94.0 | ||||
| ---------- | ||||
| * Faviconを設定できるように | ||||
| * アカウントを凍結したときすべてのフォローを解除するように | ||||
| * シェアページが機能していない問題を修正 | ||||
| * インスタンスブロックをしていてもRenote等すると取得されてしまう問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.93.1 | ||||
| ---------- | ||||
| * データのエクスポートとインポートの動作を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.93.0 | ||||
| ---------- | ||||
| * フォローリストをインポートできるように | ||||
| * embedプレイヤーを閉じれるように | ||||
| * リストをインポートしたときにプロキシアカウントがフォローするように修正 | ||||
| * Web Share Targetの動作を修正 | ||||
| * おすすめアンケートのチョイスを修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.92.4 | ||||
| ---------- | ||||
| * リストのエクスポートをできるように | ||||
| * ジョブキューウィジェットを追加 | ||||
| * URLプレビューのサムネイルが表示されないことがある問題を修正 | ||||
|  | ||||
| 10.92.3 | ||||
| ---------- | ||||
| * 管理画面の各種ジョブ数がおかしい問題を修正 | ||||
| * ジョブキューの動作を調整 | ||||
|  | ||||
| 10.92.2 | ||||
| ---------- | ||||
| * 管理画面で各種ジョブ数を一覧できるように | ||||
| * ジョブキューの動作を修正 | ||||
| * notes/children が遅い問題を修正 | ||||
|  | ||||
| 10.92.1 | ||||
| ---------- | ||||
| * アンケートの結果をリモートと同期するように | ||||
| * ジョブキューを有効に | ||||
| * 投稿の返信一覧に引用Renoteも含めるように | ||||
| * robots.txt追加 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.92.0 | ||||
| ---------- | ||||
| * Mastodonのアンケートに対応 | ||||
| * 複数回答できるアンケートを作成できるように | ||||
| * アンケートに期限を設定できるように | ||||
| * 絵文字ピッカーを改良 | ||||
| * ハッシュタグの判定を改善 | ||||
| * デッキのタグTLで別のタグをクリックしてもTLが変わらない問題を修正 | ||||
| * ユーザーサジェストで表示名が変わらない問題を修正 | ||||
| * UIのバグ修正 | ||||
| * デザインの調整 | ||||
| * など | ||||
|  | ||||
| 10.91.2 | ||||
| ---------- | ||||
| * 10.91.1 で追加した依存関係にXSS脆弱性があったので他のパッケージに差し替え | ||||
| * 初期アクセスでテーマが正しく設定されない問題を修正 | ||||
|  | ||||
| 10.91.1 | ||||
| ---------- | ||||
| * ログビューを強化 | ||||
| * テーマの切り替えをなめらかに | ||||
| * SVGの判定を修正 | ||||
|  | ||||
| 10.91.0 | ||||
| ---------- | ||||
| * ログを管理画面で見れるように | ||||
| * 文字サイズを設定できるように | ||||
| * 返信が表示されない問題を修正 | ||||
| * ユーザーページでユーザーを切り替えても前の人の情報が残る問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.90.4 | ||||
| ---------- | ||||
| * url-previewでembedプレイヤー展開をオプトインにするように | ||||
| * デザインの調整 | ||||
| * ユーザビリティの強化 | ||||
|  | ||||
| 10.90.3 | ||||
| ---------- | ||||
| * モバイルのデッキで投稿フォームウィジェットが設置できなかった問題を修正 | ||||
| * ドキュメントの強化 | ||||
| * デザインの調整 | ||||
| * ユーザビリティの強化 | ||||
|  | ||||
| 10.90.2 | ||||
| ---------- | ||||
| * アカウントが削除できない問題を修正 | ||||
| * ドキュメントの強化 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.90.1 | ||||
| ---------- | ||||
| * アカウントを作成したときに自動でホームに遷移しない問題を修正 | ||||
| * ユーザビリティの強化 | ||||
|  | ||||
| 10.90.0 | ||||
| ---------- | ||||
| * モバイル版でもデッキを使えるように | ||||
| * 公開範囲がホームの投稿はハイライトに載せないように | ||||
| * ドキュメントの強化 | ||||
| * ユーザーをリストに追加できない問題を修正 | ||||
| * UIの修正 | ||||
|  | ||||
| 10.89.1 | ||||
| ---------- | ||||
| * リアクション数を表示するように | ||||
| * モバイル版でドライブのフォルダを削除できるように | ||||
| * ドキュメントの強化 | ||||
| * プロフィールが更新できない場合がある問題を修正 | ||||
| * UIの修正 | ||||
|  | ||||
| 10.89.0 | ||||
| ---------- | ||||
| * APIのエラーの形式を統一 | ||||
| * APIドキュメント刷新 | ||||
| * /api/v1/instance/peers 復活 | ||||
| * 「返信が遷移後も残り続ける問題を修正」([9beddc9](https://github.com/syuilo/misskey/commit/9beddc941a716f1322ae0b7d71d159edd642a399)) によって遷移前に返信が表示されなくなった問題を修正 | ||||
| * デッキモードにてユーザーのプロフィールを連続で見たとき、アクティビティや画像が前のユーザーのもののまま表示される問題を修正 | ||||
|  | ||||
| 10.88.0 | ||||
| ---------- | ||||
| * アカウントの削除を試験的に実装 | ||||
| * デッキでメディア投稿のみ表示するオプションが機能していない問題を修正 | ||||
| * デッキでユーザーを表示したときにタイムラインが残存する問題を修正 | ||||
| * モバイルのユーザーページで、ユーザーAのタイムラインから他のユーザーBを選択してユーザーBのタイムラインに移動したとき、ユーザーAのタイムラインが残る問題を修正 | ||||
| * ハイライトでミュートしているユーザーの投稿が含まれる問題を修正 | ||||
| * 「みつける」でミュートしているユーザーが含まれる問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.87.5 | ||||
| ---------- | ||||
| * モバイル版でも連携サービスを表示するように | ||||
| * webfingerのacceptが反映されない問題を修正 | ||||
| * 返信が遷移後も残り続ける問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.87.4 | ||||
| ---------- | ||||
| * フォローリクエストを許可するときにエラーになる問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.87.3 | ||||
| ---------- | ||||
| * 開発モードでビルドしてもスクリプトが404になる問題を修正 | ||||
| * 拡張子判別だとアイコンやバナー設定で対応していないと表示される問題を修正 | ||||
| * フォローリクエスト数がおかしい場合の応急処置APIを追加 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.87.2 | ||||
| ---------- | ||||
| * みつけるの人気のタグを第2ソートで連合含めたユーザー数にしたりユーザーのタグ以外は除外するように | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.87.1 | ||||
| ---------- | ||||
| * ハッシュタグ検索で大文字小文字が区別されてしまう問題を修正 | ||||
|  | ||||
| 10.87.0 | ||||
| ---------- | ||||
| * ハッシュタグでユーザー検索できるように | ||||
| * Exploreページに新規ユーザー一覧を追加 | ||||
| * デッキ使用中にホーム扱いで開かれた時にタイムラインボタン等がない問題を修正 | ||||
| * デッキ使用中に / 以外でリロードした際にホームモードになる問題を修正 | ||||
|  | ||||
| 10.86.2 | ||||
| ---------- | ||||
| * 別タブでルートより下を開いたときにはデッキにしないように | ||||
| * 横のナビゲーションバーの改善 | ||||
| * MIDIファイルがオーディオ扱いになる問題を修正 | ||||
| * ミュートワードで正規表現を使えるように | ||||
| * デッキで無効になったタイムラインに警告を表示するように | ||||
| * デザインの調整 | ||||
| * その他細かな修正 | ||||
|  | ||||
| 10.86.1 | ||||
| ---------- | ||||
| * ナビゲーションバーの「ホーム」を「タイムライン」に改称 | ||||
| * モバイル版でユーザーページが二重に描画される問題を修正 | ||||
| * ユーザー一覧の「もっと読み込む」の動作がおかしい問題を修正 | ||||
| * デザインの調整 | ||||
|  | ||||
| 10.86.0 | ||||
| ---------- | ||||
| * Exploreページを実装 | ||||
|   | ||||
							
								
								
									
										173
									
								
								CONTRIBUTING.md
									
									
									
									
									
								
							
							
						
						
									
										173
									
								
								CONTRIBUTING.md
									
									
									
									
									
								
							| @@ -40,13 +40,46 @@ Stands for _**M**iss**k**ey_. | ||||
| Stands for _**S**ervice**W**orker_. | ||||
|  | ||||
| ### Nyaize | ||||
| な を にゃ にすること | ||||
| Convert な(na) to にゃ(nya) | ||||
|  | ||||
| #### Denyaize | ||||
| Nyaizeを解除すること | ||||
| Revert Nyaize | ||||
|  | ||||
| ## Code style | ||||
| ### Don't use `export default` | ||||
| ### セミコロンを省略しない | ||||
| ASI Hazardを避けるためでもある | ||||
|  | ||||
| ### 中括弧を省略しない | ||||
| Bad: | ||||
| ``` ts | ||||
| if (foo) | ||||
| 	bar; | ||||
| else | ||||
| 	baz; | ||||
| ``` | ||||
|  | ||||
| Good: | ||||
| ``` ts | ||||
| if (foo) { | ||||
| 	bar; | ||||
| } else { | ||||
| 	baz; | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ただし**`if`が一行**の時だけは省略しても良い | ||||
| Good: | ||||
| ``` ts | ||||
| if (foo) bar; | ||||
| ``` | ||||
|  | ||||
| ### `export default`を使わない | ||||
| インテリセンスと相性が悪かったりするため | ||||
|  | ||||
| 参考: | ||||
| * https://gfx.hatenablog.com/entry/2017/11/24/135343 | ||||
| * https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html | ||||
|  | ||||
| Bad: | ||||
| ``` ts | ||||
| export default function(foo: string): string { | ||||
| @@ -59,16 +92,132 @@ export function something(foo: string): string { | ||||
|  | ||||
| ## Directory structure | ||||
| ``` | ||||
| src ... ソースコード | ||||
| 	@types ... 外部ライブラリなどの型定義 | ||||
| 	prelude ... Misskeyに関係ないかつ副作用なし | ||||
| 	misc ... 副作用なしのユーティリティ処理 | ||||
| 	service ... 副作用ありの共通処理 | ||||
| 	queue ... ジョブキューとジョブ | ||||
| 	server ... Webサーバー | ||||
| 	client ... クライアント | ||||
| src ... Source code | ||||
| 	@types ... Type definitions | ||||
| 	prelude ... Independence utils for coding JavaScript without side effects | ||||
| 	misc ... Independence utils for Misskey without side effects | ||||
| 	service ... Common functions with side effects | ||||
| 	queue ... Job queues and Jobs | ||||
| 	server ... Web Server | ||||
| 	client ... Client | ||||
| 	mfm ... MFM | ||||
|  | ||||
| test ... テスト | ||||
| test ... Test code | ||||
|  | ||||
| ``` | ||||
|  | ||||
| ## Notes | ||||
| ### placeholder | ||||
| SQLをクエリビルダで組み立てる際、使用するプレースホルダは重複してはならない | ||||
| 例えば | ||||
| ``` ts | ||||
| query.andWhere(new Brackets(qb => { | ||||
| 	for (const type of ps.fileType) { | ||||
| 		qb.orWhere(`:type = ANY(note.attachedFileTypes)`, { type: type }); | ||||
| 	} | ||||
| })); | ||||
| ``` | ||||
| と書くと、ループ中で`type`というプレースホルダが複数回使われてしまいおかしくなる | ||||
| だから次のようにする必要がある | ||||
| ```ts | ||||
| query.andWhere(new Brackets(qb => { | ||||
| 	for (const type of ps.fileType) { | ||||
| 		const i = ps.fileType.indexOf(type); | ||||
| 		qb.orWhere(`:type${i} = ANY(note.attachedFileTypes)`, { [`type${i}`]: type }); | ||||
| 	} | ||||
| })); | ||||
| ``` | ||||
|  | ||||
| ### Not `null` in TypeORM | ||||
| ```ts | ||||
| const foo = await Foos.findOne({ | ||||
| 	bar: Not(null) | ||||
| }); | ||||
| ``` | ||||
| のようなクエリ(`bar`が`null`ではない)は期待通りに動作しない。 | ||||
| 次のようにします: | ||||
| ```ts | ||||
| const foo = await Foos.findOne({ | ||||
| 	bar: Not(IsNull()) | ||||
| }); | ||||
| ``` | ||||
|  | ||||
| ### `null` in SQL | ||||
| SQLを発行する際、パラメータが`null`になる可能性のある場合はSQL文を出し分けなければならない | ||||
| 例えば | ||||
| ``` ts | ||||
| query.where('file.folderId = :folderId', { folderId: ps.folderId }); | ||||
| ``` | ||||
| という処理で、`ps.folderId`が`null`だと結果的に`file.folderId = null`のようなクエリが発行されてしまい、これは正しいSQLではないので期待した結果が得られない | ||||
| だから次のようにする必要がある | ||||
| ``` ts | ||||
| if (ps.folderId) { | ||||
| 	query.where('file.folderId = :folderId', { folderId: ps.folderId }); | ||||
| } else { | ||||
| 	query.where('file.folderId IS NULL'); | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### `[]` in SQL | ||||
| SQLを発行する際、`IN`のパラメータが`[]`(空の配列)になる可能性のある場合はSQL文を出し分けなければならない | ||||
| 例えば | ||||
| ``` ts | ||||
| const users = await Users.find({ | ||||
| 	id: In(userIds) | ||||
| }); | ||||
| ``` | ||||
| という処理で、`userIds`が`[]`だと結果的に`user.id IN ()`のようなクエリが発行されてしまい、これは正しいSQLではないので期待した結果が得られない | ||||
| だから次のようにする必要がある | ||||
| ``` ts | ||||
| const users = userIds.length > 0 ? await Users.find({ | ||||
| 	id: In(userIds) | ||||
| }) : []; | ||||
| ``` | ||||
|  | ||||
| ### 配列のインデックス in SQL | ||||
| SQLでは配列のインデックスは**1始まり**。 | ||||
| `[a, b, c]`の `a`にアクセスしたいなら`[0]`ではなく`[1]`と書く | ||||
|  | ||||
| ### `undefined`にご用心 | ||||
| MongoDBの時とは違い、findOneでレコードを取得する時に対象レコードが存在しない場合 **`undefined`** が返ってくるので注意。 | ||||
| MongoDBは`null`で返してきてたので、その感覚で`if (x === null)`とか書くとバグる。代わりに`if (x == null)`と書いてください | ||||
|  | ||||
| ### 簡素な`undefined`チェック | ||||
| データベースからレコードを取得するときに、プログラムの流れ的に(ほぼ)絶対`undefined`にはならない場合でも、`undefined`チェックしないとTypeScriptに怒られます。 | ||||
| でもいちいち複数行を費やして、発生するはずのない`undefined`をチェックするのも面倒なので、`ensure`というユーティリティ関数を用意しています。 | ||||
| 例えば、 | ||||
| ``` ts | ||||
| const user = await Users.findOne(userId); | ||||
| // この時点で user の型は User | undefined | ||||
| if (user == null) { | ||||
| 	throw 'missing user'; | ||||
| } | ||||
| // この時点で user の型は User | ||||
| ``` | ||||
| という処理を`ensure`を使うと | ||||
| ``` ts | ||||
| const user = await Users.findOne(userId).then(ensure); | ||||
| // この時点で user の型は User | ||||
| ``` | ||||
| という風に書けます。 | ||||
| もちろん`ensure`内部でエラーを握りつぶすようなことはしておらず、万が一`undefined`だった場合はPromiseがRejectされ後続の処理は実行されません。 | ||||
| ``` ts | ||||
| const user = await Users.findOne(userId).then(ensure); | ||||
| // 万が一 Users.findOne の結果が undefined だったら、ensure でエラーが発生するので | ||||
| // この行に到達することは無い | ||||
| // なので、.then(ensure) は | ||||
| // if (user == null) { | ||||
| //	throw 'missing user'; | ||||
| // } | ||||
| // の糖衣構文のような扱いです | ||||
| ``` | ||||
|  | ||||
| ### Migration作成方法 | ||||
| コードの変更をした後、`ormconfig.json`(`npm run ormconfig`で生成)を用意し、 | ||||
|  | ||||
| ``` | ||||
| npm i -g ts-node | ||||
| ts-node ./node_modules/typeorm/cli.js migration:generate -n 変更の名前 | ||||
| ``` | ||||
|  | ||||
| 作成されたスクリプトは不必要な変更を含むため除去してください。 | ||||
|   | ||||
							
								
								
									
										11
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| FROM node:11-alpine AS base | ||||
| FROM node:12.1-alpine AS base | ||||
|  | ||||
| ENV NODE_ENV=production | ||||
|  | ||||
| @@ -21,21 +21,22 @@ RUN apk add --no-cache \ | ||||
|     pkgconfig \ | ||||
|     python \ | ||||
|     zlib-dev | ||||
| RUN npm i -g yarn | ||||
|  | ||||
| COPY package.json ./ | ||||
| RUN npm i | ||||
| COPY . ./ | ||||
| RUN yarn install | ||||
| RUN yarn build | ||||
| RUN npm run build | ||||
|  | ||||
| FROM base AS runner | ||||
|  | ||||
| RUN apk add --no-cache \ | ||||
|     ffmpeg \ | ||||
|     tini | ||||
| RUN npm i -g web-push | ||||
| ENTRYPOINT ["/sbin/tini", "--"] | ||||
|  | ||||
| COPY --from=builder /misskey/node_modules ./node_modules | ||||
| COPY --from=builder /misskey/built ./built | ||||
| COPY . ./ | ||||
|  | ||||
| CMD ["npm", "start"] | ||||
| CMD ["npm", "run", "migrateandstart"] | ||||
|   | ||||
							
								
								
									
										97
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										97
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| <img src="https://github.com/syuilo/misskey/blob/develop/assets/ai-orig.png?raw=true" align="right" height="320px"/> | ||||
| <a href="https://xn--931a.moe/"><img src="https://github.com/syuilo/misskey/blob/develop/assets/ai-orig.png?raw=true" align="right" height="320px"/></a> | ||||
|  | ||||
| [](https://misskey.xyz/) | ||||
| [](https://misskey.io/) | ||||
| ================================================================ | ||||
|  | ||||
| [](https://circleci.com/gh/syuilo/misskey) | ||||
| @@ -10,7 +10,7 @@ | ||||
| **A forever evolving, sophisticated microblogging platform.** | ||||
|  | ||||
| <p align="justify"> | ||||
| <a href="https://misskey.xyz">Misskey</a> is a decentralized microblogging platform born on Earth. | ||||
| <a href="https://misskey.io">Misskey</a> is a decentralized microblogging platform born on Earth. | ||||
| Since it exists within the Fediverse (a universe where various social media platforms are organized), | ||||
| it is mutually linked with other social media platforms. | ||||
| Why don't you take a short break from the hustle and bustle of the city, and dive into a new Internet? <a href="https://joinmisskey.github.io/">Find an instance!</a> | ||||
| @@ -59,7 +59,15 @@ Organize and store your files! Want to post a picture you have already uploaded? | ||||
|  | ||||
| --- | ||||
|  | ||||
| ...and more! Experience Misskey with your own eyes at [misskey.xyz](https://misskey.xyz) or join one of the [other instances](https://joinmisskey.github.io/) that are available. | ||||
| ...and more! Experience Misskey with your own eyes at [misskey.io](https://misskey.io/) or join one of the [other instances](https://joinmisskey.github.io/) that are available. | ||||
|  | ||||
| Screen shots | ||||
| ---------------------------------------------------------------- | ||||
| ### Profile page | ||||
| <img src="/assets/ss/user.jpg" width="500px"/> | ||||
|  | ||||
| ### Explore users | ||||
| <img src="/assets/ss/explore.jpg" width="500px"/> | ||||
|  | ||||
| :new: What's new | ||||
| ---------------------------------------------------------------- | ||||
| @@ -80,12 +88,14 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| 		<td><img src="https://avatars0.githubusercontent.com/u/10798641?s=460&v=4" alt="AyaMorisawa" width="100"></td> | ||||
| 		<td><img src="https://avatars1.githubusercontent.com/u/30769358?s=460&v=4" alt="mei23" width="100"></td> | ||||
| 		<td><img src="https://avatars2.githubusercontent.com/u/20679825?s=460&v=4" alt="acid-chicken" width="100"></td> | ||||
| 		<td><img src="https://avatars2.githubusercontent.com/u/6533808?s=460&v=4" alt="rinsuki" width="100"></td> | ||||
| 	</tr> | ||||
| 	<tr> | ||||
| 		<td align="center"><a href="https://github.com/syuilo">@syuilo</a></td> | ||||
| 		<td align="center"><a href="https://github.com/AyaMorisawa">@AyaMorisawa</a></td> | ||||
| 		<td align="center"><a href="https://github.com/mei23">@mei23</a></td> | ||||
| 		<td align="center"><a href="https://github.com/acid-chicken">@acid-chicken</a></td> | ||||
| 		<td align="center"><a href="https://github.com/rinsuki">@rinsuki</a></td> | ||||
| 	</tr> | ||||
| </table> | ||||
|  | ||||
| @@ -93,58 +103,79 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| ---------------------------------------------------------------- | ||||
| <!-- PATREON_START --> | ||||
| <table><tr> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12190916/fb7fa7983c14425f890369535b1506a4/1?token-time=2145916800&token-hash=WeuDzzz24cRXJogyIkU-mxARqkdyms-rcZKbO-GpGjw%3D" alt="weep" width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/12059069" alt="naga_rus" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12913507/f7181eacafe8469a93033d85f5969c29/3?token-time=2145916800&token-hash=c8HeVqLtmdgH-gSBJg8i10gmOcwllM87MDHeznl3el0%3D" alt="Melilot" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5888816/36da0f7c15954df0ab13f9abdf227f66/1.jpeg?token-time=2145916800&token-hash=at8QpJXJ8C0zINY_NmoMKv-MhXVoUK-YzTgaJPJzJYU%3D" alt="Hiroshi Seki" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12190916/fb7fa7983c14425f890369535b1506a4/3.png?token-time=2145916800&token-hash=oH_i7gJjNT7Ot6j9JiVwy7ZJIBqACVnzLqlz4YrDAZA%3D" alt="weepjp" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/19045173/cb91c0f345c24d4ebfd05f19906d5e26/1.png?token-time=2145916800&token-hash=o_zKBytJs_AxHwSYw_5R8eD0eSJe3RoTR3kR3Q0syN0%3D" alt="kiritan" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/13099460/43cecdbaa63a40d79bf50a96b9910b9d/1.jpe?token-time=2145916800&token-hash=bqwLTk0Wo0hUJJ8J5y7ii05bLzz-_CDA7Bo0Mp4RFU0%3D" alt="ne_moni" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12913507/f7181eacafe8469a93033d85f5969c29/4.jpe?token-time=2145916800&token-hash=zEyJqVM7u9d8Ri-65fJYSJcWF1jBH1nJ5a3taRzrTmw%3D" alt="Melilot" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5670915/ee175f0bfb6347ffa4ea101a8c097bff/1.jpg?token-time=2145916800&token-hash=mPLM9CA-riFHx-myr3bLZJuH2xBRHA9se5VbHhLIOuA%3D" alt="osapon" width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/16869916" alt="見当かなみ" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12999811/5f349fafcce44dd1824a8b1ebbec4564/3?token-time=2145916800&token-hash=LtV2lRi3L2jOWMLwccr9qWYfPrFlzIo2jYZHKzHEb6k%3D" alt="Xeltica" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/3384329/8b713330cb27404ea6e9fac50ff96efe/1?token-time=2145916800&token-hash=Ch3iF81ZGP0LMo894Y9ajpLisgtE91SnxtZE7fxsgrM%3D" alt="べすれい" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12021162/963128bb8d14476dbd8407943db8f31a/1?token-time=2145916800&token-hash=1FlxS9MEgmNGH_RHUVHbO5hIXB5I1z0lvA33CTvYvjA%3D" alt="gutfuckllc" width="100"></td> | ||||
| </tr><tr> | ||||
| <td><a href="https://www.patreon.com/weepjp">weep</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=12059069">naga_rus</a></td> | ||||
| <td><a href="https://www.patreon.com/rane_hs">Hiroshi Seki</a></td> | ||||
| <td><a href="https://www.patreon.com/weepjp">weepjp</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=19045173">kiritan</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=13099460">ne_moni</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=12913507">Melilot</a></td> | ||||
| <td><a href="https://www.patreon.com/osapon">osapon</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=16869916">見当かなみ</a></td> | ||||
| <td><a href="https://www.patreon.com/Xeltica">Xeltica</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=3384329">べすれい</a></td> | ||||
| <td><a href="https://www.patreon.com/gutfuckllc">gutfuckllc</a></td> | ||||
| </tr></table> | ||||
| <table><tr> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/11357794/923ce94cd8c44ba788ee931907881839/1?token-time=2145916800&token-hash=0xgcpqvFDqRcV_YIEhcPNVH7gs9sLg_BBnTJXCkN4ao%3D" alt="mydarkstar" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/18899730/6a22797f68254034a854d69ea2445fc8/1.png?token-time=2145916800&token-hash=b_uj57yxo5VzkSOUS7oXE_762dyOTB_oxzbO6lFNG3k%3D" alt="YuzuRyo61" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12021162/963128bb8d14476dbd8407943db8f31a/1.png?token-time=2145916800&token-hash=FMV7cPKBD1TU2WTbl1jg6AcdKSvTb2BSFcDhgc-EO8w%3D" alt="gutfuckllc" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/11357794/923ce94cd8c44ba788ee931907881839/1.png?token-time=2145916800&token-hash=9nEQje_eMvUjq9a7L3uBqW-MQbS-rRMaMgd7UYVoFNM%3D" alt="mydarkstar" width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/12718187" alt="Peter G." width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/16542964" alt="Takumi Sugita" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/13039004/509d0c412eb14ae08d6a812a3054f7d6/1?token-time=2145916800&token-hash=2PsbFNw0tnubZzgSXD01R6hIgncfiElG7H7HX2Y3dyo%3D" alt="nemu" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5881381/6235ca5d3fb04c8e95ef5b4ff2abcc18/3?token-time=2145916800&token-hash=9JtETp0X8gI280Ne1E8bxn6j4Lw5o2k4mJkICx97V_k%3D" alt="YUKIMOCHI" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/13039004/509d0c412eb14ae08d6a812a3054f7d6/1.jpe?token-time=2145916800&token-hash=UQRWf01TwHDV4Cls1K0YAOAjM29ssif7hLVq0ESQ0hs%3D" alt="nemu" width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/17866454" alt="sikyosyounin" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5881381/6235ca5d3fb04c8e95ef5b4ff2abcc18/3.png?token-time=2145916800&token-hash=KjfQL8nf3AIf6WqzLshBYAyX44piAqOAZiYXgZS_H6A%3D" alt="YUKIMOCHI" width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/17463605" alt="Sampot" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/17195955/be45e5e14c3e48b2bee0456c84e19df4/4?token-time=2145916800&token-hash=SbdZeN5SmsuT9stD6v0jN1z0hftg0FmRiCTxysU0Ihw%3D" alt="Damillora" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/8241184/39e18850e87a449e9c9a71acb3310ebd/3?token-time=2145916800&token-hash=gMq30aylxu5v3G8pRhWR5jeRBbYWEoRKjGbNeiCQz5g%3D" alt="Acid Chicken" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/4389829/9f709180ac714651a70f74a82f3ffdb9/2?token-time=2145916800&token-hash=zcwFxb2zopzWwksKVU1YpfAEjsl4yKT02aQ6yiAFRiQ%3D" alt="natalie" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/13034746/c711c7f58e204ecfbc2fd646bc8a4eee/1?token-time=2145916800&token-hash=5T8XcaAf9Zyzfg3QubR06s_kJZkArVEM2dwObrBVAU4%3D" alt="Hiratake" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/19356899/496b4681d33b4520bd7688e0fd19c04d/2.jpeg?token-time=2145916800&token-hash=_sTj3dUBOhn9qwiJ7F19Qd-yWWfUqJC_0jG1h0agEqQ%3D" alt="sheeta.s" width="100"></td> | ||||
| </tr><tr> | ||||
| <td><a href="https://www.patreon.com/Yuzulia">YuzuRyo61</a></td> | ||||
| <td><a href="https://www.patreon.com/gutfuckllc">gutfuckllc</a></td> | ||||
| <td><a href="https://www.patreon.com/mydarkstar">mydarkstar</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=12718187">Peter G.</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=16542964">Takumi Sugita</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=13039004">nemu</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=17866454">sikyosyounin</a></td> | ||||
| <td><a href="https://www.patreon.com/yukimochi">YUKIMOCHI</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=17463605">Sampot</a></td> | ||||
| <td><a href="https://www.patreon.com/damillora">Damillora</a></td> | ||||
| <td><a href="https://www.patreon.com/acid_chicken">Acid Chicken</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=4389829">natalie</a></td> | ||||
| <td><a href="https://www.patreon.com/hiratake">Hiratake</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=19356899">sheeta.s</a></td> | ||||
| </tr></table> | ||||
| <table><tr> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/4503830/ccf2cc867ea64de0b524bb2e24b9a1cb/1?token-time=2145916800&token-hash=Ksk_2l3gjPDbnzMUOCSW1E-hdPJsNs2tSR4_RAakRK8%3D" alt="dansup" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/619786/32cf01444db24e578cd1982c197f6fc6/1?token-time=2145916800&token-hash=CXe9AqlZy9AsYfiWd3OBYVOzvODoN47Litz0Tu4BFpU%3D" alt="Gargron" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5731881/4b6038e6cda34c04b83a5fcce3806a93/1?token-time=2145916800&token-hash=xhR1n6NAAyEb-IUXLD6_dshkFa3mefU5ZZuk1L8qKTs%3D" alt="Nokotaro Takeda" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12531784/93a45137841849329ba692da92ac7c60/1?token-time=2145916800&token-hash=uR-48MQ0A4j0irQSrCAQZJ-sJUSs_Fkihlg3-l59b7c%3D" alt="Takashi Shibuya" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/13737140/1adf7835017d479280d90fe8d30aade2/1.png?token-time=2145916800&token-hash=0pdle8h5pDZrww0BDOjdz6zO-HudeGTh36a3qi1biVU%3D" alt="Satsuki Yanagi" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/17880724/311738c8a48f4a6b9443c2445a75adde/1.jpe?token-time=2145916800&token-hash=CPxGQhKIlEaa6WUcgbyHixyKEhakiw9RFdOhsIJBQ_o%3D" alt="takimura" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/17195955/be45e5e14c3e48b2bee0456c84e19df4/4.jpe?token-time=2145916800&token-hash=UslrPVM-8TXOe8AapuNiaFYjcIJgPNcU-fKpGbfGJNI%3D" alt="Damillora" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/16900731/83884b38afc74d4cbe83c30a13b10edd/1.png?token-time=2145916800&token-hash=R5Tog8RWg0rguRoCIoir3lThokrdPvs8Utfikhc0nhY%3D" alt="Atsuko Tominaga" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/4389829/9f709180ac714651a70f74a82f3ffdb9/3.png?token-time=2145916800&token-hash=FTm3WVom4dJ9NwWMU4OpCL_8Yc13WiwEbKrDPyTZTPs%3D" alt="natalie" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/13034746/c711c7f58e204ecfbc2fd646bc8a4eee/1.jpe?token-time=2145916800&token-hash=EWxXhVbZYH7KB4IDT3joc8TbIg8zPO40x1r5IDn3R7c%3D" alt="Hiratake" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5923936/2a743cbfbff946c2af3f09026047c0da/2.png?token-time=2145916800&token-hash=h6yphW1qnM0n_NOWaf8qtszMRLXEwIxfk5beu4RxdT0%3D" alt="noellabo" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/2384390/5681180e1efb46a8b28e0e8d4c8b9037/1.jpg?token-time=2145916800&token-hash=SJcMy-Q1BcS940-LFUVOMfR7-5SgrzsEQGhYb3yowFk%3D" alt="CG" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/18072312/98e894d960314fa7bc236a72a39488fe/1.jpe?token-time=2145916800&token-hash=qA8j97lIZNc-74AuZ0p4F3ms6sKPeKjtNt2vEuwpsyo%3D" alt="Hekovic" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/10789744/97175095d8f04c0f86225ff47cb98d40/1.jpeg?token-time=2145916800&token-hash=l4AoMR7Nj7K4yAHrkrk2hAoggPkbSPm12m1nmbe9Pb8%3D" alt="Naoki Hirayama" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/4503830/ccf2cc867ea64de0b524bb2e24b9a1cb/1.jpeg?token-time=2145916800&token-hash=L55UhJ0rcuNAH3w_ryeeGN4hC6taoOixyAhraEi0bzw%3D" alt="dansup" width="100"></td> | ||||
| </tr><tr> | ||||
| <td><a href="https://www.patreon.com/user?u=13737140">Satsuki Yanagi</a></td> | ||||
| <td><a href="https://www.patreon.com/takimura">takimura</a></td> | ||||
| <td><a href="https://www.patreon.com/damillora">Damillora</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=16900731">Atsuko Tominaga</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=4389829">natalie</a></td> | ||||
| <td><a href="https://www.patreon.com/hiratake">Hiratake</a></td> | ||||
| <td><a href="https://www.patreon.com/noellabo">noellabo</a></td> | ||||
| <td><a href="https://www.patreon.com/Corset">CG</a></td> | ||||
| <td><a href="https://www.patreon.com/hekovic">Hekovic</a></td> | ||||
| <td><a href="https://www.patreon.com/spinlock">Naoki Hirayama</a></td> | ||||
| <td><a href="https://www.patreon.com/dansup">dansup</a></td> | ||||
| </tr></table> | ||||
| <table><tr> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/619786/32cf01444db24e578cd1982c197f6fc6/1.jpeg?token-time=2145916800&token-hash=d8jBQLMOHD87KtXs5C9fk1o58DMF73pQ-dYH3uZJPBE%3D" alt="Gargron" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5731881/4b6038e6cda34c04b83a5fcce3806a93/1.png?token-time=2145916800&token-hash=hBayGfOmQH3kRMdNnDe4oCZD_9fsJWSt29xXR3KRMVk%3D" alt="Nokotaro Takeda" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12531784/93a45137841849329ba692da92ac7c60/1.jpeg?token-time=2145916800&token-hash=vGe7wXGqmA8Q7m-kDNb6fyGdwk-Dxk4F-ut8ZZu51RM%3D" alt="Takashi Shibuya" width="100"></td> | ||||
| </tr><tr> | ||||
| <td><a href="https://www.patreon.com/mastodon">Gargron</a></td> | ||||
| <td><a href="https://www.patreon.com/takenoko">Nokotaro Takeda</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=12531784">Takashi Shibuya</a></td> | ||||
| </tr></table> | ||||
|  | ||||
| **Last updated:** Fri, 15 Feb 2019 19:12:06 UTC | ||||
| **Last updated:** Sun, 19 May 2019 02:37:06 UTC | ||||
| <!-- PATREON_END --> | ||||
|  | ||||
| :four_leaf_clover: Copyright | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								assets/api-doc.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								assets/api-doc.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 5.4 KiB | 
							
								
								
									
										4
									
								
								assets/robots.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								assets/robots.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| user-agent: * | ||||
| allow: / | ||||
|  | ||||
| # todo: sitemap | ||||
							
								
								
									
										
											BIN
										
									
								
								assets/ss/explore.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								assets/ss/explore.jpg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 238 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/ss/user.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								assets/ss/user.jpg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 148 KiB | 
| @@ -1,9 +0,0 @@ | ||||
| { | ||||
| 	'targets': [ | ||||
| 		{ | ||||
| 			'target_name': 'crypto_key', | ||||
| 			'sources': ['src/crypto_key.cc'], | ||||
| 			'include_dirs': ['<!(node -e "require(\'nan\')")'] | ||||
| 		} | ||||
| 	] | ||||
| } | ||||
| @@ -1,23 +0,0 @@ | ||||
| const mongo = require('mongodb'); | ||||
| const User = require('../built/models/user').default; | ||||
|  | ||||
| const args = process.argv.slice(2); | ||||
|  | ||||
| const user = args[0]; | ||||
|  | ||||
| const q = user.startsWith('@') ? { | ||||
| 	username: user.split('@')[1], | ||||
| 	host: user.split('@')[2] || null | ||||
| } : { _id: new mongo.ObjectID(user) }; | ||||
|  | ||||
| console.log(`Mark as admin ${user}...`); | ||||
|  | ||||
| User.update(q, { | ||||
| 	$set: { | ||||
| 		isAdmin: true | ||||
| 	} | ||||
| }).then(() => { | ||||
| 	console.log(`Done ${user}`); | ||||
| }, e => { | ||||
| 	console.error(e); | ||||
| }); | ||||
| @@ -1,57 +0,0 @@ | ||||
| // for Node.js interpret | ||||
|  | ||||
| const chalk = require('chalk'); | ||||
| const sequential = require('promise-sequential'); | ||||
|  | ||||
| const { default: User } = require('../../built/models/user'); | ||||
| const { default: DriveFile } = require('../../built/models/drive-file'); | ||||
|  | ||||
| async function main() { | ||||
| 	const promiseGens = []; | ||||
|  | ||||
| 	const count = await DriveFile.count({}); | ||||
|  | ||||
| 	let prev; | ||||
|  | ||||
| 	for (let i = 0; i < count; i++) { | ||||
| 		promiseGens.push(() => { | ||||
| 			const promise = new Promise(async (res, rej) => { | ||||
| 				const file = await DriveFile.findOne(prev ? { | ||||
| 					_id: { $gt: prev._id } | ||||
| 				} : {}, { | ||||
| 					sort: { | ||||
| 						_id: 1 | ||||
| 					} | ||||
| 				}); | ||||
|  | ||||
| 				prev = file; | ||||
|  | ||||
| 				const user = await User.findOne({ _id: file.metadata.userId }); | ||||
|  | ||||
| 				DriveFile.update({ | ||||
| 					_id: file._id | ||||
| 				}, { | ||||
| 					$set: { | ||||
| 						'metadata._user': { | ||||
| 							host: user.host | ||||
| 						} | ||||
| 					} | ||||
| 				}).then(() => { | ||||
| 					res([i, file]); | ||||
| 				}).catch(rej); | ||||
| 			}); | ||||
|  | ||||
| 			promise.then(([i, file]) => { | ||||
| 				console.log(chalk`{gray ${i}} {green done: {bold ${file._id}} ${file.filename}}`); | ||||
| 			}); | ||||
|  | ||||
| 			return promise; | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| 	return await sequential(promiseGens); | ||||
| } | ||||
|  | ||||
| main().then(() => { | ||||
| 	console.log('ALL DONE'); | ||||
| }).catch(console.error); | ||||
| @@ -1,71 +0,0 @@ | ||||
| // for Node.js interpret | ||||
|  | ||||
| const chalk = require('chalk'); | ||||
| const sequential = require('promise-sequential'); | ||||
|  | ||||
| const { default: User } = require('../../built/models/user'); | ||||
| const { default: DriveFile } = require('../../built/models/drive-file'); | ||||
|  | ||||
| async function main() { | ||||
| 	const promiseGens = []; | ||||
|  | ||||
| 	const count = await User.count({}); | ||||
|  | ||||
| 	let prev; | ||||
|  | ||||
| 	for (let i = 0; i < count; i++) { | ||||
| 		promiseGens.push(() => { | ||||
| 			const promise = new Promise(async (res, rej) => { | ||||
| 				const user = await User.findOne(prev ? { | ||||
| 					_id: { $gt: prev._id } | ||||
| 				} : {}, { | ||||
| 					sort: { | ||||
| 						_id: 1 | ||||
| 					} | ||||
| 				}); | ||||
|  | ||||
| 				prev = user; | ||||
|  | ||||
| 				const set = {}; | ||||
|  | ||||
| 				if (user.avatarId != null) { | ||||
| 					const file = await DriveFile.findOne({ _id: user.avatarId }); | ||||
|  | ||||
| 					if (file && file.metadata.properties.avgColor) { | ||||
| 						set.avatarColor = file.metadata.properties.avgColor; | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				if (user.bannerId != null) { | ||||
| 					const file = await DriveFile.findOne({ _id: user.bannerId }); | ||||
|  | ||||
| 					if (file && file.metadata.properties.avgColor) { | ||||
| 						set.bannerColor = file.metadata.properties.avgColor; | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				if (Object.keys(set).length === 0) return res([i, user]); | ||||
|  | ||||
| 				User.update({ | ||||
| 					_id: user._id | ||||
| 				}, { | ||||
| 					$set: set | ||||
| 				}).then(() => { | ||||
| 					res([i, user]); | ||||
| 				}).catch(rej); | ||||
| 			}); | ||||
|  | ||||
| 			promise.then(([i, user]) => { | ||||
| 				console.log(chalk`{gray ${i}} {green done: {bold ${user._id}} @${user.username}}`); | ||||
| 			}); | ||||
|  | ||||
| 			return promise; | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| 	return await sequential(promiseGens); | ||||
| } | ||||
|  | ||||
| main().then(() => { | ||||
| 	console.log('ALL DONE'); | ||||
| }).catch(console.error); | ||||
| @@ -1,9 +0,0 @@ | ||||
| const { default: DriveFile } = require('../../built/models/drive-file'); | ||||
|  | ||||
| DriveFile.update({}, { | ||||
| 	$rename: { | ||||
| 		'metadata.isMetaOnly': 'metadata.withoutChunks' | ||||
| 	} | ||||
| }, { | ||||
| 	multi: true | ||||
| }); | ||||
| @@ -1,134 +0,0 @@ | ||||
| const { default: Stats } = require('../../built/models/stats'); | ||||
| const { default: User } = require('../../built/models/user'); | ||||
| const { default: Note } = require('../../built/models/note'); | ||||
| const { default: DriveFile } = require('../../built/models/drive-file'); | ||||
|  | ||||
| const now = new Date(); | ||||
| const y = now.getFullYear(); | ||||
| const m = now.getMonth(); | ||||
| const d = now.getDate(); | ||||
| const today = new Date(y, m, d); | ||||
|  | ||||
| async function main() { | ||||
| 	const localUsersCount = await User.count({ | ||||
| 		host: null | ||||
| 	}); | ||||
|  | ||||
| 	const remoteUsersCount = await User.count({ | ||||
| 		host: { $ne: null } | ||||
| 	}); | ||||
|  | ||||
| 	const localNotesCount = await Note.count({ | ||||
| 		'_user.host': null | ||||
| 	}); | ||||
|  | ||||
| 	const remoteNotesCount = await Note.count({ | ||||
| 		'_user.host': { $ne: null } | ||||
| 	}); | ||||
|  | ||||
| 	const localDriveFilesCount = await DriveFile.count({ | ||||
| 		'metadata._user.host': null | ||||
| 	}); | ||||
|  | ||||
| 	const remoteDriveFilesCount = await DriveFile.count({ | ||||
| 		'metadata._user.host': { $ne: null } | ||||
| 	}); | ||||
|  | ||||
| 	const localDriveFilesSize = await DriveFile | ||||
| 		.aggregate([{ | ||||
| 			$match: { | ||||
| 				'metadata._user.host': null, | ||||
| 				'metadata.deletedAt': { $exists: false } | ||||
| 			} | ||||
| 		}, { | ||||
| 			$project: { | ||||
| 				length: true | ||||
| 			} | ||||
| 		}, { | ||||
| 			$group: { | ||||
| 				_id: null, | ||||
| 				usage: { $sum: '$length' } | ||||
| 			} | ||||
| 		}]) | ||||
| 		.then(aggregates => { | ||||
| 			if (aggregates.length > 0) { | ||||
| 				return aggregates[0].usage; | ||||
| 			} | ||||
| 			return 0; | ||||
| 		}); | ||||
|  | ||||
| 	const remoteDriveFilesSize = await DriveFile | ||||
| 		.aggregate([{ | ||||
| 			$match: { | ||||
| 				'metadata._user.host': { $ne: null }, | ||||
| 				'metadata.deletedAt': { $exists: false } | ||||
| 			} | ||||
| 		}, { | ||||
| 			$project: { | ||||
| 				length: true | ||||
| 			} | ||||
| 		}, { | ||||
| 			$group: { | ||||
| 				_id: null, | ||||
| 				usage: { $sum: '$length' } | ||||
| 			} | ||||
| 		}]) | ||||
| 		.then(aggregates => { | ||||
| 			if (aggregates.length > 0) { | ||||
| 				return aggregates[0].usage; | ||||
| 			} | ||||
| 			return 0; | ||||
| 		}); | ||||
|  | ||||
| 	await Stats.insert({ | ||||
| 		date: today, | ||||
| 		users: { | ||||
| 			local: { | ||||
| 				total: localUsersCount, | ||||
| 				diff: 0 | ||||
| 			}, | ||||
| 			remote: { | ||||
| 				total: remoteUsersCount, | ||||
| 				diff: 0 | ||||
| 			} | ||||
| 		}, | ||||
| 		notes: { | ||||
| 			local: { | ||||
| 				total: localNotesCount, | ||||
| 				diff: 0, | ||||
| 				diffs: { | ||||
| 					normal: 0, | ||||
| 					reply: 0, | ||||
| 					renote: 0 | ||||
| 				} | ||||
| 			}, | ||||
| 			remote: { | ||||
| 				total: remoteNotesCount, | ||||
| 				diff: 0, | ||||
| 				diffs: { | ||||
| 					normal: 0, | ||||
| 					reply: 0, | ||||
| 					renote: 0 | ||||
| 				} | ||||
| 			} | ||||
| 		}, | ||||
| 		drive: { | ||||
| 			local: { | ||||
| 				totalCount: localDriveFilesCount, | ||||
| 				totalSize: localDriveFilesSize, | ||||
| 				diffCount: 0, | ||||
| 				diffSize: 0 | ||||
| 			}, | ||||
| 			remote: { | ||||
| 				totalCount: remoteDriveFilesCount, | ||||
| 				totalSize: remoteDriveFilesSize, | ||||
| 				diffCount: 0, | ||||
| 				diffSize: 0 | ||||
| 			} | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	console.log('done'); | ||||
| } | ||||
|  | ||||
| main(); | ||||
| @@ -1,144 +0,0 @@ | ||||
| const { default: Stats } = require('../../built/models/stats'); | ||||
| const { default: User } = require('../../built/models/user'); | ||||
| const { default: Note } = require('../../built/models/note'); | ||||
| const { default: DriveFile } = require('../../built/models/drive-file'); | ||||
|  | ||||
| const now = new Date(); | ||||
| const y = now.getFullYear(); | ||||
| const m = now.getMonth(); | ||||
| const d = now.getDate(); | ||||
| const h = now.getHours(); | ||||
| const date = new Date(y, m, d, h); | ||||
|  | ||||
| async function main() { | ||||
| 	await Stats.update({}, { | ||||
| 		$set: { | ||||
| 			span: 'day' | ||||
| 		} | ||||
| 	}, { | ||||
| 		multi: true | ||||
| 	}); | ||||
|  | ||||
| 	const localUsersCount = await User.count({ | ||||
| 		host: null | ||||
| 	}); | ||||
|  | ||||
| 	const remoteUsersCount = await User.count({ | ||||
| 		host: { $ne: null } | ||||
| 	}); | ||||
|  | ||||
| 	const localNotesCount = await Note.count({ | ||||
| 		'_user.host': null | ||||
| 	}); | ||||
|  | ||||
| 	const remoteNotesCount = await Note.count({ | ||||
| 		'_user.host': { $ne: null } | ||||
| 	}); | ||||
|  | ||||
| 	const localDriveFilesCount = await DriveFile.count({ | ||||
| 		'metadata._user.host': null | ||||
| 	}); | ||||
|  | ||||
| 	const remoteDriveFilesCount = await DriveFile.count({ | ||||
| 		'metadata._user.host': { $ne: null } | ||||
| 	}); | ||||
|  | ||||
| 	const localDriveFilesSize = await DriveFile | ||||
| 		.aggregate([{ | ||||
| 			$match: { | ||||
| 				'metadata._user.host': null, | ||||
| 				'metadata.deletedAt': { $exists: false } | ||||
| 			} | ||||
| 		}, { | ||||
| 			$project: { | ||||
| 				length: true | ||||
| 			} | ||||
| 		}, { | ||||
| 			$group: { | ||||
| 				_id: null, | ||||
| 				usage: { $sum: '$length' } | ||||
| 			} | ||||
| 		}]) | ||||
| 		.then(aggregates => { | ||||
| 			if (aggregates.length > 0) { | ||||
| 				return aggregates[0].usage; | ||||
| 			} | ||||
| 			return 0; | ||||
| 		}); | ||||
|  | ||||
| 	const remoteDriveFilesSize = await DriveFile | ||||
| 		.aggregate([{ | ||||
| 			$match: { | ||||
| 				'metadata._user.host': { $ne: null }, | ||||
| 				'metadata.deletedAt': { $exists: false } | ||||
| 			} | ||||
| 		}, { | ||||
| 			$project: { | ||||
| 				length: true | ||||
| 			} | ||||
| 		}, { | ||||
| 			$group: { | ||||
| 				_id: null, | ||||
| 				usage: { $sum: '$length' } | ||||
| 			} | ||||
| 		}]) | ||||
| 		.then(aggregates => { | ||||
| 			if (aggregates.length > 0) { | ||||
| 				return aggregates[0].usage; | ||||
| 			} | ||||
| 			return 0; | ||||
| 		}); | ||||
|  | ||||
| 	await Stats.insert({ | ||||
| 		date: date, | ||||
| 		span: 'hour', | ||||
| 		users: { | ||||
| 			local: { | ||||
| 				total: localUsersCount, | ||||
| 				diff: 0 | ||||
| 			}, | ||||
| 			remote: { | ||||
| 				total: remoteUsersCount, | ||||
| 				diff: 0 | ||||
| 			} | ||||
| 		}, | ||||
| 		notes: { | ||||
| 			local: { | ||||
| 				total: localNotesCount, | ||||
| 				diff: 0, | ||||
| 				diffs: { | ||||
| 					normal: 0, | ||||
| 					reply: 0, | ||||
| 					renote: 0 | ||||
| 				} | ||||
| 			}, | ||||
| 			remote: { | ||||
| 				total: remoteNotesCount, | ||||
| 				diff: 0, | ||||
| 				diffs: { | ||||
| 					normal: 0, | ||||
| 					reply: 0, | ||||
| 					renote: 0 | ||||
| 				} | ||||
| 			} | ||||
| 		}, | ||||
| 		drive: { | ||||
| 			local: { | ||||
| 				totalCount: localDriveFilesCount, | ||||
| 				totalSize: localDriveFilesSize, | ||||
| 				diffCount: 0, | ||||
| 				diffSize: 0 | ||||
| 			}, | ||||
| 			remote: { | ||||
| 				totalCount: remoteDriveFilesCount, | ||||
| 				totalSize: remoteDriveFilesSize, | ||||
| 				diffCount: 0, | ||||
| 				diffSize: 0 | ||||
| 			} | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	console.log('done'); | ||||
| } | ||||
|  | ||||
| main(); | ||||
| @@ -5,8 +5,8 @@ services: | ||||
|     build: . | ||||
|     restart: always | ||||
|     links: | ||||
|       - mongo | ||||
| #      - redis | ||||
|       - db | ||||
|       - redis | ||||
| #      - es | ||||
|     ports: | ||||
|       - "127.0.0.1:3000:3000" | ||||
| @@ -14,26 +14,23 @@ services: | ||||
|       - internal_network | ||||
|       - external_network | ||||
|  | ||||
| #  redis: | ||||
| #    restart: always | ||||
| #    image: redis:4.0-alpine | ||||
| #    networks: | ||||
| #      - internal_network | ||||
| ### Uncomment to enable Redis persistance | ||||
| ##    volumes: | ||||
| ##      - ./redis:/data | ||||
|  | ||||
|   mongo: | ||||
|   redis: | ||||
|     restart: always | ||||
|     image: mongo:4.1 | ||||
|     image: redis:4.0-alpine | ||||
|     networks: | ||||
|       - internal_network | ||||
|     environment: | ||||
|       MONGO_INITDB_DATABASE: "misskey" | ||||
|     volumes: | ||||
|       - ./.config/mongo_initdb.js:/docker-entrypoint-initdb.d/mongo_initdb.js:ro | ||||
| ### Uncomment to enable MongoDB persistance | ||||
| #      - ./mongo:/data | ||||
|       - ./redis:/data | ||||
|  | ||||
|   db: | ||||
|     restart: always | ||||
|     image: postgres:11.2-alpine | ||||
|     networks: | ||||
|       - internal_network | ||||
|     env_file: | ||||
|       - .config/docker.env | ||||
|     volumes: | ||||
|       - ./db:/var/lib/postgresql/data | ||||
|  | ||||
| #  es: | ||||
| #    restart: always | ||||
| @@ -42,9 +39,8 @@ services: | ||||
| #      - "ES_JAVA_OPTS=-Xms512m -Xmx512m" | ||||
| #    networks: | ||||
| #      - internal_network | ||||
| #### Uncomment to enable ES persistence | ||||
| ##    volumes: | ||||
| ##      - ./elasticsearch:/usr/share/elasticsearch/data | ||||
| #    volumes: | ||||
| #      - ./elasticsearch:/usr/share/elasticsearch/data | ||||
|  | ||||
| networks: | ||||
|   internal_network: | ||||
|   | ||||
| @@ -1,22 +0,0 @@ | ||||
| Comment faire une sauvegarde de votre Misskey ? | ||||
| ========================== | ||||
|  | ||||
| Assurez-vous d'avoir installé **mongodb-tools**. | ||||
|  | ||||
| --- | ||||
|  | ||||
| Dans votre terminal : | ||||
| ``` shell | ||||
| $ mongodump --archive=db-backup -u <VotreNomdUtilisateur> -p <VotreMotDePasse> | ||||
| ``` | ||||
|  | ||||
| Pour plus de détails, merci de consulter [la documentation de mongodump](https://docs.mongodb.com/manual/reference/program/mongodump/). | ||||
|  | ||||
| Restauration | ||||
| ------- | ||||
|  | ||||
| ``` shell | ||||
| $ mongorestore --archive=db-backup | ||||
| ``` | ||||
|  | ||||
| Pour plus de détails, merci de consulter [la documentation de mongorestore](https://docs.mongodb.com/manual/reference/program/mongorestore/). | ||||
| @@ -1,22 +0,0 @@ | ||||
| How to backup your Misskey | ||||
| ========================== | ||||
|  | ||||
| Make sure **mongodb-tools** installed. | ||||
|  | ||||
| --- | ||||
|  | ||||
| In your shell: | ||||
| ``` shell | ||||
| $ mongodump --archive=db-backup -u <YourUserName> -p <YourPassword> | ||||
| ``` | ||||
|  | ||||
| For details, please see [mongodump docs](https://docs.mongodb.com/manual/reference/program/mongodump/). | ||||
|  | ||||
| Restore | ||||
| ------- | ||||
|  | ||||
| ``` shell | ||||
| $ mongorestore --archive=db-backup | ||||
| ``` | ||||
|  | ||||
| For details, please see [mongorestore docs](https://docs.mongodb.com/manual/reference/program/mongorestore/). | ||||
| @@ -9,15 +9,51 @@ This guide describes how to install and setup Misskey with Docker. | ||||
|  | ||||
| *1.* Download Misskey | ||||
| ---------------------------------------------------------------- | ||||
| 1. `git clone -b master git://github.com/syuilo/misskey.git` Clone Misskey repository's master branch. | ||||
| 2. `cd misskey` Move to misskey directory. | ||||
| 3. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` Checkout to the [latest release](https://github.com/syuilo/misskey/releases/latest) tag. | ||||
| 1. Clone Misskey repository's master branch. | ||||
|  | ||||
| 	`git clone -b master git://github.com/syuilo/misskey.git` | ||||
|  | ||||
| 2. Move to misskey directory. | ||||
|  | ||||
| 	`cd misskey` | ||||
|  | ||||
| 3. Checkout to the [latest release](https://github.com/syuilo/misskey/releases/latest) tag. | ||||
|  | ||||
| 	`git checkout master` | ||||
|  | ||||
| *2.* Configure Misskey | ||||
| ---------------------------------------------------------------- | ||||
| 1. `cp .config/example.yml .config/default.yml` Copy the `.config/example.yml` and rename it to `default.yml`. | ||||
| 2. `cp .config/mongo_initdb_example.js .config/mongo_initdb.js` Copy the `.config/mongo_initdb_example.js` and rename it to `mongo_initdb.js`. | ||||
| 3. Edit `default.yml` and `mongo_initdb.js`. | ||||
|  | ||||
| Create configuration files with following: | ||||
|  | ||||
| ```bash | ||||
| cd .config | ||||
| cp example.yml default.yml | ||||
| cp docker_example.env docker.env | ||||
| ``` | ||||
|  | ||||
| ### `default.yml` | ||||
|  | ||||
| Edit this file the same as non-Docker environment.   | ||||
| However hostname of Postgresql, Redis and Elasticsearch are not `localhost`, they are set in `docker-compose.yml`.   | ||||
| The following is default hostname: | ||||
|  | ||||
| | Service       | Hostname | | ||||
| |---------------|----------| | ||||
| | Postgresql    | `db`     | | ||||
| | Redis         | `redis`  | | ||||
| | Elasticsearch | `es`     | | ||||
|  | ||||
| ### `docker.env` | ||||
|  | ||||
| Configure Postgresql in this file.   | ||||
| The minimum required settings are: | ||||
|  | ||||
| | name                | Description   | | ||||
| |---------------------|---------------| | ||||
| | `POSTGRES_PASSWORD` | Password      | | ||||
| | `POSTGRES_USER`     | Username      | | ||||
| | `POSTGRES_DB`       | Database name | | ||||
|  | ||||
| *3.* Configure Docker | ||||
| ---------------------------------------------------------------- | ||||
| @@ -29,7 +65,13 @@ Build misskey with the following: | ||||
|  | ||||
| `docker-compose build` | ||||
|  | ||||
| *5.* That is it. | ||||
| *5.* Init DB | ||||
| ---------------------------------------------------------------- | ||||
| ``` shell | ||||
| docker-compose run --rm web npm run init | ||||
| ``` | ||||
|  | ||||
| *6.* That is it. | ||||
| ---------------------------------------------------------------- | ||||
| Well done! Now you have an environment to run Misskey. | ||||
|  | ||||
| @@ -37,9 +79,9 @@ Well done! Now you have an environment to run Misskey. | ||||
| Just `docker-compose up -d`. GLHF! | ||||
|  | ||||
| ### How to update your Misskey server to the latest version | ||||
| 1. `git fetch` | ||||
| 2. `git stash` | ||||
| 3. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` | ||||
| 1. `git stash` | ||||
| 2. `git checkout master` | ||||
| 3. `git pull` | ||||
| 4. `git stash pop` | ||||
| 5. `docker-compose build` | ||||
| 6. Check [ChangeLog](../CHANGELOG.md) for migration information | ||||
|   | ||||
| @@ -10,9 +10,17 @@ Ce guide explique comment installer et configurer Misskey avec Docker. | ||||
|  | ||||
| *1.* Télécharger Misskey | ||||
| ---------------------------------------------------------------- | ||||
| 1. `git clone -b master git://github.com/syuilo/misskey.git` Clone le dépôt de Misskey sur la branche master. | ||||
| 2. `cd misskey` Naviguez dans le dossier du dépôt. | ||||
| 3. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` Checkout sur le tag de la [dernière version](https://github.com/syuilo/misskey/releases/latest). | ||||
| 1. Clone le dépôt de Misskey sur la branche master. | ||||
|  | ||||
| 	`git clone -b master git://github.com/syuilo/misskey.git` | ||||
|  | ||||
| 2. Naviguez dans le dossier du dépôt. | ||||
|  | ||||
| 	`cd misskey` | ||||
|  | ||||
| 3. Checkout sur le tag de la [dernière version](https://github.com/syuilo/misskey/releases/latest). | ||||
|  | ||||
| 	`git checkout master` | ||||
|  | ||||
| *2.* Configuration de Misskey | ||||
| ---------------------------------------------------------------- | ||||
| @@ -38,9 +46,9 @@ Parfait, Vous avez un environnement prêt pour démarrer Misskey. | ||||
| Utilisez la commande `docker-compose up -d`. GLHF! | ||||
|  | ||||
| ### How to update your Misskey server to the latest version | ||||
| 1. `git fetch` | ||||
| 2. `git stash` | ||||
| 3. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` | ||||
| 1. `git stash` | ||||
| 2. `git checkout master` | ||||
| 3. `git pull` | ||||
| 4. `git stash pop` | ||||
| 5. `docker-compose build` | ||||
| 6. Consultez le [ChangeLog](../CHANGELOG.md) pour avoir les éventuelles informations de migration | ||||
| @@ -52,14 +60,28 @@ Utilisez la commande `docker-compose up -d`. GLHF! | ||||
| ### Configuration d'ElasticSearch (pour la fonction de recherche) | ||||
| *1.* Préparation de l'environnement | ||||
| ---------------------------------------------------------------- | ||||
| 1. `mkdir elasticsearch && chown 1000:1000 elasticsearch` Permet de créer le dossier d'accueil de la base ElasticSearch aves les bons droits | ||||
| 2. `sysctl -w vm.max_map_count=262144` Augmente la valeur max du paramètre map_count du système (valeur minimum pour pouvoir lancer ES) | ||||
| 1. Permet de créer le dossier d'accueil de la base ElasticSearch aves les bons droits | ||||
|  | ||||
| 	`mkdir elasticsearch && chown 1000:1000 elasticsearch` | ||||
|  | ||||
| 2. Augmente la valeur max du paramètre map_count du système (valeur minimum pour pouvoir lancer ES) | ||||
|  | ||||
| 	`sysctl -w vm.max_map_count=262144` | ||||
|  | ||||
| *2.* Après lancement du docker-compose, initialisation de la base ElasticSearch | ||||
| ---------------------------------------------------------------- | ||||
| 1. `docker-compose -it web /bin/sh` Connexion dans le conteneur web | ||||
| 2. `apk add curl` Ajout du paquet curl | ||||
| 3. `curl -X PUT "es:9200/misskey" -H 'Content-Type: application/json' -d'{ "settings" : { "index" : { } }}'` Création de la base ES | ||||
| 1. Connexion dans le conteneur web | ||||
|  | ||||
| 	`docker-compose -it web /bin/sh` | ||||
|  | ||||
| 2. Ajout du paquet curl | ||||
|  | ||||
| 	`apk add curl` | ||||
|  | ||||
| 3. Création de la base ES | ||||
|  | ||||
| 	`curl -X PUT "es:9200/misskey" -H 'Content-Type: application/json' -d'{ "settings" : { "index" : { } }}'` | ||||
|  | ||||
| 4. `exit` | ||||
|  | ||||
| ---------------------------------------------------------------- | ||||
|   | ||||
| @@ -9,15 +9,51 @@ Dockerを使ったMisskey構築方法 | ||||
|  | ||||
| *1.* Misskeyのダウンロード | ||||
| ---------------------------------------------------------------- | ||||
| 1. `git clone -b master git://github.com/syuilo/misskey.git` masterブランチからMisskeyレポジトリをクローン | ||||
| 2. `cd misskey` misskeyディレクトリに移動 | ||||
| 3. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` [最新のリリース](https://github.com/syuilo/misskey/releases/latest)を確認 | ||||
| 1. masterブランチからMisskeyレポジトリをクローン | ||||
|  | ||||
| *2.* 設定ファイルを作成する | ||||
| 	`git clone -b master git://github.com/syuilo/misskey.git` | ||||
|  | ||||
| 2. misskeyディレクトリに移動 | ||||
|  | ||||
| 	`cd misskey` | ||||
|  | ||||
| 3. [最新のリリース](https://github.com/syuilo/misskey/releases/latest)を確認 | ||||
|  | ||||
| 	`git checkout master` | ||||
|  | ||||
| *2.* 設定ファイルの作成と編集 | ||||
| ---------------------------------------------------------------- | ||||
| 1. `cp .config/example.yml .config/default.yml` `.config/example.yml`をコピーし名前を`default.yml`にする | ||||
| 2. `cp .config/mongo_initdb_example.js .config/mongo_initdb.js` `.config/mongo_initdb_example.js`をコピーし名前を`mongo_initdb.js`にする | ||||
| 3. `default.yml`と`mongo_initdb.js`を編集する | ||||
|  | ||||
| 下記コマンドで設定ファイルを作成してください。 | ||||
|  | ||||
| ```bash | ||||
| cd .config | ||||
| cp example.yml default.yml | ||||
| cp docker_example.env docker.env | ||||
| ``` | ||||
|  | ||||
| ### `default.yml`の編集 | ||||
|  | ||||
| 非Docker環境と同じ様に編集してください。   | ||||
| ただし、Postgresql、RedisとElasticsearchのホストは`localhost`ではなく、`docker-compose.yml`で設定されたサービス名になっています。   | ||||
| 標準設定では次の通りです。 | ||||
|  | ||||
| | サービス       | ホスト名 | | ||||
| |---------------|---------| | ||||
| | Postgresql    |`db`     | | ||||
| | Redis         |`redis`  | | ||||
| | Elasticsearch |`es`     | | ||||
|  | ||||
| ### `docker.env`の編集 | ||||
|  | ||||
| このファイルはPostgresqlの設定を記述します。   | ||||
| 最低限記述する必要がある設定は次の通りです。 | ||||
|  | ||||
| | 設定                 | 内容         | | ||||
| |---------------------|--------------| | ||||
| | `POSTGRES_PASSWORD` | パスワード    | | ||||
| | `POSTGRES_USER`     | ユーザー名    | | ||||
| | `POSTGRES_DB`       | データベース名 | | ||||
|  | ||||
| *3.* Dockerの設定 | ||||
| ---------------------------------------------------------------- | ||||
| @@ -29,7 +65,13 @@ Dockerを使ったMisskey構築方法 | ||||
|  | ||||
| `docker-compose build` | ||||
|  | ||||
| *5.* 以上です! | ||||
| *5.* データベースを初期化 | ||||
| ---------------------------------------------------------------- | ||||
| ``` shell | ||||
| docker-compose run --rm web npm run init | ||||
| ``` | ||||
|  | ||||
| *6.* 以上です! | ||||
| ---------------------------------------------------------------- | ||||
| お疲れ様でした。これでMisskeyを動かす準備は整いました。 | ||||
|  | ||||
| @@ -37,9 +79,9 @@ Dockerを使ったMisskey構築方法 | ||||
| `docker-compose up -d`するだけです。GLHF! | ||||
|  | ||||
| ### Misskeyを最新バージョンにアップデートする方法: | ||||
| 1. `git fetch` | ||||
| 2. `git stash` | ||||
| 3. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` | ||||
| 1. `git stash` | ||||
| 2. `git checkout master` | ||||
| 3. `git pull` | ||||
| 4. `git stash pop` | ||||
| 5. `docker-compose build` | ||||
| 6. [ChangeLog](../CHANGELOG.md)でマイグレーション情報を確認する | ||||
|   | ||||
| @@ -29,7 +29,7 @@ server { | ||||
|     listen [::]:443 http2; | ||||
|     server_name example.tld; | ||||
|     ssl on; | ||||
| 		ssl_session_cache shared:ssl_session_cache:10m; | ||||
|     ssl_session_cache shared:ssl_session_cache:10m; | ||||
|  | ||||
|     # To use Let's Encrypt certificate | ||||
|     ssl_certificate     /etc/letsencrypt/live/example.tld/fullchain.pem; | ||||
|   | ||||
							
								
								
									
										118
									
								
								docs/setup.en.md
									
									
									
									
									
								
							
							
						
						
									
										118
									
								
								docs/setup.en.md
									
									
									
									
									
								
							| @@ -22,47 +22,50 @@ adduser --disabled-password --disabled-login misskey | ||||
| Please install and setup these softwares: | ||||
|  | ||||
| #### Dependencies :package: | ||||
| * **[Node.js](https://nodejs.org/en/)** >= 10.0.0 | ||||
| * **[MongoDB](https://www.mongodb.com/)** >= 3.6 | ||||
| * **[Node.js](https://nodejs.org/en/)** >= 11.7.0 | ||||
| * **[PostgreSQL](https://www.postgresql.org/)** >= 10 | ||||
| * **[Redis](https://redis.io/)** | ||||
|  | ||||
| ##### Optional | ||||
| * [Redis](https://redis.io/) | ||||
|   * Redis is optional, but we strongly recommended to install it | ||||
| * [Elasticsearch](https://www.elastic.co/) - required to enable the search feature | ||||
| * [FFmpeg](https://www.ffmpeg.org/) | ||||
|  | ||||
| *3.* Setup MongoDB | ||||
| *3.* Install Misskey | ||||
| ---------------------------------------------------------------- | ||||
| As root: | ||||
| 1. `mongo` Go to the mongo shell | ||||
| 2. `use misskey` Use the misskey database | ||||
| 3. `db.createUser( { user: "misskey", pwd: "<password>", roles: [ { role: "readWrite", db: "misskey" } ] } )` Create the misskey user. | ||||
| 4. `exit` You're done! | ||||
| 1. Connect to misskey user. | ||||
|  | ||||
| *4.* Install Misskey | ||||
| ---------------------------------------------------------------- | ||||
| 1. `su - misskey` Connect to misskey user. | ||||
| 2. `git clone -b master git://github.com/syuilo/misskey.git` Clone the misskey repo from master branch. | ||||
| 3. `cd misskey` Navigate to misskey directory | ||||
| 4. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` Checkout to the [latest release](https://github.com/syuilo/misskey/releases/latest) | ||||
| 5. `npm install` Install misskey dependencies. | ||||
| 	`su - misskey` | ||||
|  | ||||
| *5.* Configure Misskey | ||||
| 2. Clone the misskey repo from master branch. | ||||
|  | ||||
| 	`git clone -b master git://github.com/syuilo/misskey.git` | ||||
|  | ||||
| 3. Navigate to misskey directory | ||||
|  | ||||
| 	`cd misskey` | ||||
|  | ||||
| 4. Checkout to the [latest release](https://github.com/syuilo/misskey/releases/latest) | ||||
|  | ||||
| 	`git checkout master` | ||||
|  | ||||
| 5. Install misskey dependencies. | ||||
|  | ||||
| 	`npm install` | ||||
|  | ||||
| *4.* Configure Misskey | ||||
| ---------------------------------------------------------------- | ||||
| 1. `cp .config/example.yml .config/default.yml` Copy the `.config/example.yml` and rename it to `default.yml`. | ||||
| 1. Copy the `.config/example.yml` and rename it to `default.yml`. | ||||
|  | ||||
| 	`cp .config/example.yml .config/default.yml` | ||||
|  | ||||
| 2. Edit `default.yml` | ||||
|  | ||||
| *6.* Build Misskey | ||||
| *5.* Build Misskey | ||||
| ---------------------------------------------------------------- | ||||
|  | ||||
| Before build, you need to set `NODE_ENV` to `production`. like this: | ||||
| * Linux: `export NODE_ENV=production` | ||||
| * Windows (PowerShell): `$env:NODE_ENV="production"` | ||||
| * Windows (CMD): `set NODE_ENV=production` | ||||
|  | ||||
| Build misskey with the following: | ||||
|  | ||||
| `npm run build` | ||||
| `NODE_ENV=production npm run build` | ||||
|  | ||||
| If you're on Debian, you will need to install the `build-essential`, `python` package. | ||||
|  | ||||
| @@ -71,49 +74,64 @@ If you're still encountering errors about some modules, use node-gyp: | ||||
| 1. `npm install -g node-gyp` | ||||
| 2. `node-gyp configure` | ||||
| 3. `node-gyp build` | ||||
| 4. `npm run build` | ||||
| 4. `NODE_ENV=production npm run build` | ||||
|  | ||||
| *6.* Init DB | ||||
| ---------------------------------------------------------------- | ||||
| ``` shell | ||||
| npm run init | ||||
| ``` | ||||
|  | ||||
| *7.* That is it. | ||||
| ---------------------------------------------------------------- | ||||
| Well done! Now, you have an environment that run to Misskey. | ||||
|  | ||||
| ### Launch normally | ||||
| Just `npm start`. GLHF! | ||||
| Just `NODE_ENV=production npm start`. GLHF! | ||||
|  | ||||
| ### Launch with systemd | ||||
|  | ||||
| 1. Create a systemd service here: `/etc/systemd/system/misskey.service` | ||||
| 1. Create a systemd service here | ||||
|  | ||||
| 	`/etc/systemd/system/misskey.service` | ||||
|  | ||||
| 2. Edit it, and paste this and save: | ||||
|  | ||||
| ``` | ||||
| [Unit] | ||||
| Description=Misskey daemon | ||||
| 	``` | ||||
| 	[Unit] | ||||
| 	Description=Misskey daemon | ||||
|  | ||||
| [Service] | ||||
| Type=simple | ||||
| User=misskey | ||||
| ExecStart=/usr/bin/npm start | ||||
| WorkingDirectory=/home/misskey/misskey | ||||
| TimeoutSec=60 | ||||
| StandardOutput=syslog | ||||
| StandardError=syslog | ||||
| SyslogIdentifier=misskey | ||||
| Restart=always | ||||
| 	[Service] | ||||
| 	Type=simple | ||||
| 	User=misskey | ||||
| 	ExecStart=/usr/bin/npm start | ||||
| 	WorkingDirectory=/home/misskey/misskey | ||||
| 	Environment="NODE_ENV=production" | ||||
| 	TimeoutSec=60 | ||||
| 	StandardOutput=syslog | ||||
| 	StandardError=syslog | ||||
| 	SyslogIdentifier=misskey | ||||
| 	Restart=always | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
| ``` | ||||
| 	[Install] | ||||
| 	WantedBy=multi-user.target | ||||
| 	``` | ||||
|  | ||||
| 3. `systemctl daemon-reload ; systemctl enable misskey` Reload systemd and enable the misskey service. | ||||
| 4. `systemctl start misskey` Start the misskey service. | ||||
| 3. Reload systemd and enable the misskey service. | ||||
|  | ||||
| 	`systemctl daemon-reload ; systemctl enable misskey` | ||||
|  | ||||
| 4. Start the misskey service. | ||||
|  | ||||
| 	`systemctl start misskey` | ||||
|  | ||||
| You can check if the service is running with `systemctl status misskey`. | ||||
|  | ||||
| ### How to update your Misskey server to the latest version | ||||
| 1. `git fetch` | ||||
| 2. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` | ||||
| 1. `git checkout master` | ||||
| 2. `git pull` | ||||
| 3. `npm install` | ||||
| 4. `npm run build` | ||||
| 4. `NODE_ENV=production npm run build` | ||||
| 5. Check [ChangeLog](../CHANGELOG.md) for migration information | ||||
| 6. Restart your Misskey process to apply changes | ||||
| 7. Enjoy | ||||
|   | ||||
							
								
								
									
										109
									
								
								docs/setup.fr.md
									
									
									
									
									
								
							
							
						
						
									
										109
									
								
								docs/setup.fr.md
									
									
									
									
									
								
							| @@ -22,42 +22,50 @@ adduser --disabled-password --disabled-login misskey | ||||
| Installez les paquets suivants : | ||||
|  | ||||
| #### Dépendences :package: | ||||
| * **[Node.js](https://nodejs.org/en/)** >= 10.0.0 | ||||
| * **[MongoDB](https://www.mongodb.com/)** >= 3.6 | ||||
| * **[Node.js](https://nodejs.org/en/)** >= 11.7.0 | ||||
| * **[PostgreSQL](https://www.postgresql.org/)** >= 10 | ||||
| * **[Redis](https://redis.io/)** | ||||
|  | ||||
| ##### Optionnels | ||||
| * [Redis](https://redis.io/) | ||||
|   * Redis est optionnel mais nous vous recommandons vivement de l'installer | ||||
| * [Elasticsearch](https://www.elastic.co/) - requis pour pouvoir activer la fonctionnalité de recherche | ||||
| * [FFmpeg](https://www.ffmpeg.org/) | ||||
|  | ||||
| *3.* Paramètrage de MongoDB | ||||
| *3.* Installation de Misskey | ||||
| ---------------------------------------------------------------- | ||||
| En root : | ||||
| 1. `mongo` Ouvrez le shell mongo | ||||
| 2. `use misskey` Utilisez la base de données misskey | ||||
| 3. `db.createUser( { user: "misskey", pwd: "<password>", roles: [ { role: "readWrite", db: "misskey" } ] } )` Créez l'utilisateur misskey. | ||||
| 4. `exit` Vous avez terminé ! | ||||
| 1. Basculez vers l'utilisateur misskey. | ||||
|  | ||||
| *4.* Installation de Misskey | ||||
| ---------------------------------------------------------------- | ||||
| 1. `su - misskey` Basculez vers l'utilisateur misskey. | ||||
| 2. `git clone -b master git://github.com/syuilo/misskey.git` Clonez la branche master du dépôt misskey. | ||||
| 3. `cd misskey` Accédez au dossier misskey. | ||||
| 4. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` Checkout sur le tag de la [version la plus récente](https://github.com/syuilo/misskey/releases/latest) | ||||
| 5. `npm install` Installez les dépendances de misskey. | ||||
| 	`su - misskey` | ||||
|  | ||||
| *5.* Création du fichier de configuration | ||||
| 2. Clonez la branche master du dépôt misskey. | ||||
|  | ||||
| 	`git clone -b master git://github.com/syuilo/misskey.git` | ||||
|  | ||||
| 3. Accédez au dossier misskey. | ||||
|  | ||||
| 	`cd misskey` | ||||
|  | ||||
| 4. Checkout sur le tag de la [version la plus récente](https://github.com/syuilo/misskey/releases/latest) | ||||
|  | ||||
| 	`git checkout master` | ||||
|   | ||||
| 5. Installez les dépendances de misskey. | ||||
|  | ||||
| 	`npm install` | ||||
|  | ||||
| *4.* Création du fichier de configuration | ||||
| ---------------------------------------------------------------- | ||||
| 1. `cp .config/example.yml .config/default.yml` Copiez le fichier `.config/example.yml` et renommez-le`default.yml`. | ||||
| 1. Copiez le fichier `.config/example.yml` et renommez-le`default.yml`. | ||||
|  | ||||
| 	`cp .config/example.yml .config/default.yml` | ||||
|  | ||||
| 2. Editez le fichier `default.yml` | ||||
|  | ||||
| *6.* Construction de Misskey | ||||
| *5.* Construction de Misskey | ||||
| ---------------------------------------------------------------- | ||||
|  | ||||
| Construisez Misskey comme ceci : | ||||
|  | ||||
| `npm run build` | ||||
| `NODE_ENV=production npm run build` | ||||
|  | ||||
| Si vous êtes sous Debian, vous serez amené à installer les paquets `build-essential` et `python`. | ||||
|  | ||||
| @@ -66,49 +74,58 @@ Si vous rencontrez des erreurs concernant certains modules, utilisez node-gyp: | ||||
| 1. `npm install -g node-gyp` | ||||
| 2. `node-gyp configure` | ||||
| 3. `node-gyp build` | ||||
| 4. `npm run build` | ||||
| 4. `NODE_ENV=production npm run build` | ||||
|  | ||||
| *7.* C'est tout. | ||||
| *6.* C'est tout. | ||||
| ---------------------------------------------------------------- | ||||
| Excellent ! Maintenant, vous avez un environnement prêt pour lancer Misskey | ||||
|  | ||||
| ### Lancement conventionnel | ||||
| Lancez tout simplement `npm start`. Bonne chance et amusez-vous bien ! | ||||
| Lancez tout simplement `NODE_ENV=production npm start`. Bonne chance et amusez-vous bien ! | ||||
|  | ||||
| ### Démarrage avec systemd | ||||
|  | ||||
| 1. Créez un service systemd sur : `/etc/systemd/system/misskey.service` | ||||
| 1. Créez un service systemd sur | ||||
|  | ||||
| 	`/etc/systemd/system/misskey.service` | ||||
|  | ||||
| 2. Editez-le puis copiez et coller ceci dans le fichier : | ||||
|  | ||||
| ``` | ||||
| [Unit] | ||||
| Description=Misskey daemon | ||||
| 	``` | ||||
| 	[Unit] | ||||
| 	Description=Misskey daemon | ||||
|  | ||||
| [Service] | ||||
| Type=simple | ||||
| User=misskey | ||||
| ExecStart=/usr/bin/npm start | ||||
| WorkingDirectory=/home/misskey/misskey | ||||
| TimeoutSec=60 | ||||
| StandardOutput=syslog | ||||
| StandardError=syslog | ||||
| SyslogIdentifier=misskey | ||||
| Restart=always | ||||
| 	[Service] | ||||
| 	Type=simple | ||||
| 	User=misskey | ||||
| 	ExecStart=/usr/bin/npm start | ||||
| 	WorkingDirectory=/home/misskey/misskey | ||||
| 	Environment="NODE_ENV=production" | ||||
| 	TimeoutSec=60 | ||||
| 	StandardOutput=syslog | ||||
| 	StandardError=syslog | ||||
| 	SyslogIdentifier=misskey | ||||
| 	Restart=always | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
| ``` | ||||
| 	[Install] | ||||
| 	WantedBy=multi-user.target | ||||
| 	``` | ||||
|  | ||||
| 3. `systemctl daemon-reload ; systemctl enable misskey` Redémarre systemd et active le service misskey. | ||||
| 4. `systemctl start misskey` Démarre le service misskey. | ||||
| 3. Redémarre systemd et active le service misskey. | ||||
|  | ||||
| 	`systemctl daemon-reload ; systemctl enable misskey` | ||||
|  | ||||
| 4. Démarre le service misskey. | ||||
|  | ||||
| 	`systemctl start misskey` | ||||
|  | ||||
| Vous pouvez vérifier si le service a démarré en utilisant la commande `systemctl status misskey`. | ||||
|  | ||||
| ### Méthode de mise à jour vers la plus récente version de Misskey | ||||
| 1. `git fetch` | ||||
| 2. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` | ||||
| 1. `git checkout master` | ||||
| 2. `git pull` | ||||
| 3. `npm install` | ||||
| 4. `npm run build` | ||||
| 4. `NODE_ENV=production npm run build` | ||||
| 5. Consultez [ChangeLog](../CHANGELOG.md) pour les information de migration. | ||||
|  | ||||
| ---------------------------------------------------------------- | ||||
|   | ||||
							
								
								
									
										129
									
								
								docs/setup.ja.md
									
									
									
									
									
								
							
							
						
						
									
										129
									
								
								docs/setup.ja.md
									
									
									
									
									
								
							| @@ -22,54 +22,51 @@ adduser --disabled-password --disabled-login misskey | ||||
| これらのソフトウェアをインストール・設定してください: | ||||
|  | ||||
| #### 依存関係 :package: | ||||
| * **[Node.js](https://nodejs.org/en/)** (10.0.0以上) | ||||
| * **[MongoDB](https://www.mongodb.com/)** (3.6以上) | ||||
| * **[Node.js](https://nodejs.org/en/)** (11.7.0以上) | ||||
| * **[PostgreSQL](https://www.postgresql.org/)** (10以上) | ||||
| * **[Redis](https://redis.io/)** | ||||
|  | ||||
| ##### オプション | ||||
| * [Redis](https://redis.io/) | ||||
| 	* Redisはオプションですが、インストールすることを強く推奨します。 | ||||
| 	* インストールしなくていいのは、あなたのインスタンスが自分専用のときだけとお考えください。 | ||||
| 	* 具体的には、Redisをインストールしないと、次の事が出来なくなります: | ||||
| 		* Misskeyプロセスを複数起動しての負荷分散 | ||||
| 		* レートリミット | ||||
| 		* ジョブキュー | ||||
| 		* Twitter連携 | ||||
| * [Elasticsearch](https://www.elastic.co/) | ||||
| 	* 検索機能を有効にするためにはインストールが必要です。 | ||||
| * [FFmpeg](https://www.ffmpeg.org/) | ||||
|  | ||||
| *3.* MongoDBの設定 | ||||
| *3.* Misskeyのインストール | ||||
| ---------------------------------------------------------------- | ||||
| ルートで: | ||||
| 1. `mongo` mongoシェルを起動 | ||||
| 2. `use misskey` misskeyデータベースを使用 | ||||
| 3. `db.createUser( { user: "misskey", pwd: "<password>", roles: [ { role: "readWrite", db: "misskey" } ] } )` misskeyユーザーを作成 | ||||
| 4. `exit` mongoシェルを終了 | ||||
| 1. misskeyユーザーを使用 | ||||
|  | ||||
| *4.* Misskeyのインストール | ||||
| ---------------------------------------------------------------- | ||||
| 1. `su - misskey` misskeyユーザーを使用 | ||||
| 2. `git clone -b master git://github.com/syuilo/misskey.git` masterブランチからMisskeyレポジトリをクローン | ||||
| 3. `cd misskey` misskeyディレクトリに移動 | ||||
| 4. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` [最新のリリース](https://github.com/syuilo/misskey/releases/latest)を確認 | ||||
| 5. `npm install` Misskeyの依存パッケージをインストール | ||||
| 	`su - misskey` | ||||
|  | ||||
| *5.* 設定ファイルを作成する | ||||
| 2. masterブランチからMisskeyレポジトリをクローン | ||||
|  | ||||
| 	`git clone -b master git://github.com/syuilo/misskey.git` | ||||
|  | ||||
| 3. misskeyディレクトリに移動 | ||||
|  | ||||
| 	`cd misskey` | ||||
|  | ||||
| 4. [最新のリリース](https://github.com/syuilo/misskey/releases/latest)を確認 | ||||
|  | ||||
| 	`git checkout master` | ||||
|  | ||||
| 5. Misskeyの依存パッケージをインストール | ||||
|  | ||||
| 	`npm install` | ||||
|  | ||||
| *4.* 設定ファイルを作成する | ||||
| ---------------------------------------------------------------- | ||||
| 1. `cp .config/example.yml .config/default.yml` `.config/example.yml`をコピーし名前を`default.yml`にする。 | ||||
| 1. `.config/example.yml`をコピーし名前を`default.yml`にする。 | ||||
|  | ||||
| 	`cp .config/example.yml .config/default.yml` | ||||
|  | ||||
| 2. `default.yml` を編集する。 | ||||
|  | ||||
| *6.* Misskeyのビルド | ||||
| *5.* Misskeyのビルド | ||||
| ---------------------------------------------------------------- | ||||
|  | ||||
| ビルドする前に、`NODE_ENV`を`production`にする必要があります。例: | ||||
| * Linux: `export NODE_ENV=production` | ||||
| * Windows (PowerShell): `$env:NODE_ENV="production"` | ||||
| * Windows (CMD): `set NODE_ENV=production` | ||||
|  | ||||
| 次のコマンドでMisskeyをビルドしてください: | ||||
|  | ||||
| `npm run build` | ||||
| `NODE_ENV=production npm run build` | ||||
|  | ||||
| Debianをお使いであれば、`build-essential`パッケージをインストールする必要があります。 | ||||
|  | ||||
| @@ -77,52 +74,68 @@ Debianをお使いであれば、`build-essential`パッケージをインスト | ||||
| 1. `npm install -g node-gyp` | ||||
| 2. `node-gyp configure` | ||||
| 3. `node-gyp build` | ||||
| 4. `npm run build` | ||||
| 4. `NODE_ENV=production npm run build` | ||||
|  | ||||
| *6.* データベースを初期化 | ||||
| ---------------------------------------------------------------- | ||||
| ``` shell | ||||
| npm run init | ||||
| ``` | ||||
|  | ||||
| *7.* 以上です! | ||||
| ---------------------------------------------------------------- | ||||
| お疲れ様でした。これでMisskeyを動かす準備は整いました。 | ||||
|  | ||||
| ### 通常起動 | ||||
| `npm start`するだけです。GLHF! | ||||
| `NODE_ENV=production npm start`するだけです。GLHF! | ||||
|  | ||||
| ### systemdを用いた起動 | ||||
| 1. systemdサービスのファイルを作成: `/etc/systemd/system/misskey.service` | ||||
| 1. systemdサービスのファイルを作成 | ||||
|  | ||||
| 	`/etc/systemd/system/misskey.service` | ||||
|  | ||||
| 2. エディタで開き、以下のコードを貼り付けて保存: | ||||
|  | ||||
| ``` | ||||
| [Unit] | ||||
| Description=Misskey daemon | ||||
| 	``` | ||||
| 	[Unit] | ||||
| 	Description=Misskey daemon | ||||
|  | ||||
| [Service] | ||||
| Type=simple | ||||
| User=misskey | ||||
| ExecStart=/usr/bin/npm start | ||||
| WorkingDirectory=/home/misskey/misskey | ||||
| TimeoutSec=60 | ||||
| StandardOutput=syslog | ||||
| StandardError=syslog | ||||
| SyslogIdentifier=misskey | ||||
| Restart=always | ||||
| 	[Service] | ||||
| 	Type=simple | ||||
| 	User=misskey | ||||
| 	ExecStart=/usr/bin/npm start | ||||
| 	WorkingDirectory=/home/misskey/misskey | ||||
| 	Environment="NODE_ENV=production" | ||||
| 	TimeoutSec=60 | ||||
| 	StandardOutput=syslog | ||||
| 	StandardError=syslog | ||||
| 	SyslogIdentifier=misskey | ||||
| 	Restart=always | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
| ``` | ||||
| CentOSで1024以下のポートを使用してMisskeyを使用する場合は`ExecStart=/usr/bin/sudo /usr/bin/npm start`に変更する必要があります。 | ||||
| 	[Install] | ||||
| 	WantedBy=multi-user.target | ||||
| 	``` | ||||
|  | ||||
| 3. `systemctl daemon-reload ; systemctl enable misskey` systemdを再読み込みしmisskeyサービスを有効化 | ||||
| 4. `systemctl start misskey` misskeyサービスの起動 | ||||
| 	CentOSで1024以下のポートを使用してMisskeyを使用する場合は`ExecStart=/usr/bin/sudo /usr/bin/npm start`に変更する必要があります。 | ||||
|  | ||||
| 3. systemdを再読み込みしmisskeyサービスを有効化 | ||||
|  | ||||
| 	`systemctl daemon-reload ; systemctl enable misskey` | ||||
|  | ||||
| 4. misskeyサービスの起動 | ||||
|  | ||||
| 	`systemctl start misskey` | ||||
|  | ||||
| `systemctl status misskey`と入力すると、サービスの状態を調べることができます。 | ||||
|  | ||||
| ### Misskeyを最新バージョンにアップデートする方法: | ||||
| 1. `git fetch` | ||||
| 2. `git checkout $(git tag -l | grep -v 'rc[0-9]*$' | sort -V | tail -n 1)` | ||||
| 1. `git checkout master` | ||||
| 2. `git pull` | ||||
| 3. `npm install` | ||||
| 4. `npm run build` | ||||
| 4. `NODE_ENV=production npm run build` | ||||
| 5. [ChangeLog](../CHANGELOG.md)でマイグレーション情報を確認する | ||||
|  | ||||
| なにか問題が発生した場合は、`npm run clean`すると直る場合があります。 | ||||
| なにか問題が発生した場合は、`npm run clean`または`npm run cleanall`すると直る場合があります。 | ||||
|  | ||||
| ---------------------------------------------------------------- | ||||
|  | ||||
|   | ||||
							
								
								
									
										11
									
								
								gulpfile.ts
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								gulpfile.ts
									
									
									
									
									
								
							| @@ -5,7 +5,6 @@ | ||||
| import * as gulp from 'gulp'; | ||||
| import * as gutil from 'gulp-util'; | ||||
| import * as ts from 'gulp-typescript'; | ||||
| const yaml = require('gulp-yaml'); | ||||
| const sourcemaps = require('gulp-sourcemaps'); | ||||
| import tslint from 'gulp-tslint'; | ||||
| const cssnano = require('gulp-cssnano'); | ||||
| @@ -50,7 +49,6 @@ gulp.task('build:copy:views', () => | ||||
|  | ||||
| gulp.task('build:copy', gulp.parallel('build:copy:views', () => | ||||
| 	gulp.src([ | ||||
| 		'./build/Release/crypto_key.node', | ||||
| 		'./src/const.json', | ||||
| 		'./src/server/web/views/**/*', | ||||
| 		'./src/**/assets/**/*', | ||||
| @@ -121,17 +119,11 @@ gulp.task('copy:client', () => | ||||
| 		]) | ||||
| 			.pipe(isProduction ? (imagemin as any)() : gutil.noop()) | ||||
| 			.pipe(rename(path => { | ||||
| 				path.dirname = path.dirname.replace('assets', '.'); | ||||
| 				path.dirname = path.dirname!.replace('assets', '.'); | ||||
| 			})) | ||||
| 			.pipe(gulp.dest('./built/client/assets/')) | ||||
| ); | ||||
|  | ||||
| gulp.task('locales', () => | ||||
| 	gulp.src('./locales/*.yml') | ||||
| 		.pipe(yaml({ schema: 'DEFAULT_SAFE_SCHEMA' })) | ||||
| 		.pipe(gulp.dest('./built/client/assets/locales/')) | ||||
| ); | ||||
|  | ||||
| gulp.task('doc', () => | ||||
| 	gulp.src('./src/docs/**/*.styl') | ||||
| 		.pipe(stylus()) | ||||
| @@ -149,7 +141,6 @@ gulp.task('build', gulp.parallel( | ||||
| 	'build:ts', | ||||
| 	'build:copy', | ||||
| 	'build:client', | ||||
| 	'locales', | ||||
| 	'doc' | ||||
| )); | ||||
|  | ||||
|   | ||||
							
								
								
									
										1686
									
								
								locales/ca-ES.yml
									
									
									
									
									
								
							
							
						
						
									
										1686
									
								
								locales/ca-ES.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1248
									
								
								locales/cs-CZ.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1248
									
								
								locales/cs-CZ.yml
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1896
									
								
								locales/da-DK.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1896
									
								
								locales/da-DK.yml
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1403
									
								
								locales/de-DE.yml
									
									
									
									
									
								
							
							
						
						
									
										1403
									
								
								locales/de-DE.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1345
									
								
								locales/es-ES.yml
									
									
									
									
									
								
							
							
						
						
									
										1345
									
								
								locales/es-ES.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -5,9 +5,50 @@ | ||||
| const fs = require('fs'); | ||||
| const yaml = require('js-yaml'); | ||||
|  | ||||
| const langs = ['de-DE', 'en-US', 'fr-FR', 'ja-JP', 'ja-KS', 'pl-PL', 'es-ES', 'nl-NL', 'zh-CN', 'ko-KR']; | ||||
| const merge = (...args) => args.reduce((a, c) => ({ | ||||
| 	...a, | ||||
| 	...c, | ||||
| 	...Object.entries(a) | ||||
| 		.filter(([k]) => c && typeof c[k] === 'object') | ||||
| 		.reduce((a, [k, v]) => (a[k] = merge(v, c[k]), a), {}) | ||||
| }), {}); | ||||
|  | ||||
| const loadLocale = lang => yaml.safeLoad(fs.readFileSync(`${__dirname}/${lang}.yml`, 'utf-8')); | ||||
| const locales = langs.map(lang => ({ [lang]: loadLocale(lang) })); | ||||
| const languages = [ | ||||
| 	'cs-CZ', | ||||
| 	'da-DK', | ||||
| 	'de-DE', | ||||
| 	'en-US', | ||||
| 	'es-ES', | ||||
| 	'fr-FR', | ||||
| 	'ja-JP', | ||||
| 	'ja-KS', | ||||
| 	'ko-KR', | ||||
| 	'nl-NL', | ||||
| 	'pl-PL', | ||||
| 	'zh-CN', | ||||
| 	'zh-TW', | ||||
| ]; | ||||
|  | ||||
| module.exports = locales.reduce((a, b) => ({ ...a, ...b })); | ||||
| const primaries = { | ||||
| 	'en': 'US', | ||||
| 	'ja': 'JP', | ||||
| 	'zh': 'CN', | ||||
| }; | ||||
|  | ||||
| const locales = languages.reduce((a, c) => (a[c] = yaml.safeLoad(fs.readFileSync(`${__dirname}/${c}.yml`, 'utf-8')) || {}, a), {}); | ||||
|  | ||||
| module.exports = Object.entries(locales) | ||||
| 	.reduce((a, [k ,v]) => (a[k] = (() => { | ||||
| 		const [lang] = k.split('-'); | ||||
| 		switch (k) { | ||||
| 			case 'ja-JP': return v; | ||||
| 			case 'ja-KS': | ||||
| 			case 'en-US': return merge(locales['ja-JP'], v); | ||||
| 			default: return merge( | ||||
| 				locales['ja-JP'], | ||||
| 				locales['en-US'], | ||||
| 				locales[`${lang}-${primaries[lang]}`] || {}, | ||||
| 				v | ||||
| 			); | ||||
| 		} | ||||
| 	})(), a), {}); | ||||
|   | ||||
							
								
								
									
										1683
									
								
								locales/it-IT.yml
									
									
									
									
									
								
							
							
						
						
									
										1683
									
								
								locales/it-IT.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1348
									
								
								locales/nl-NL.yml
									
									
									
									
									
								
							
							
						
						
									
										1348
									
								
								locales/nl-NL.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1404
									
								
								locales/no-NO.yml
									
									
									
									
									
								
							
							
						
						
									
										1404
									
								
								locales/no-NO.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1555
									
								
								locales/pt-PT.yml
									
									
									
									
									
								
							
							
						
						
									
										1555
									
								
								locales/pt-PT.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1570
									
								
								locales/ru-RU.yml
									
									
									
									
									
								
							
							
						
						
									
										1570
									
								
								locales/ru-RU.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										90
									
								
								locales/zh-TW.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								locales/zh-TW.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | ||||
| --- | ||||
| meta: | ||||
|   lang: "中文(繁体)" | ||||
| common: | ||||
|   intro: | ||||
|     title: "什麽是 Misskey 呢?" | ||||
|     rich-contents: "發佈" | ||||
|     reaction: "回應" | ||||
|     drive: "雲端硬碟" | ||||
|   adblock: | ||||
|     detected: "請禁用廣告封鎖器" | ||||
|   close: "關閉" | ||||
|   enter-password: "請輸入密碼" | ||||
|   2fa: "雙重身份驗證" | ||||
|   dark-mode: "夜間模式" | ||||
|   signup: "註冊" | ||||
|   signout: "登出" | ||||
|   notification: | ||||
|     reversi-invited: "您已被邀請加入壹場遊戲" | ||||
|     reversi-invited-by: "來自{}的邀請" | ||||
|     notified-by: "來自{}的邀請" | ||||
|   time: | ||||
|     future: "未來" | ||||
|     just_now: "剛剛" | ||||
|   drive: "雲端硬碟" | ||||
|   weekday: | ||||
|     sunday: "週日" | ||||
|     monday: "週一" | ||||
|     tuesday: "週二" | ||||
|     wednesday: "週三" | ||||
|     thursday: "週四" | ||||
|     friday: "週五" | ||||
|     saturday: "週六" | ||||
|   reactions: | ||||
|     like: "贊" | ||||
|     love: "喜歡" | ||||
|     congrats: "恭喜" | ||||
|   _settings: | ||||
|     password: "密碼" | ||||
|     font-size: "字體大小" | ||||
|     font-size-x-small: "小" | ||||
|     font-size-small: "較小" | ||||
|     deck-column-width-wide: "寬" | ||||
|     timeline: "時間軸" | ||||
| common/views/components/connect-failed.troubleshooter.vue: | ||||
|   flush: "清除快取" | ||||
| common/views/components/theme.vue: | ||||
|   light-themes: "淺色主題" | ||||
|   dark-themes: "深色主題" | ||||
|   install-a-theme: "安裝主題" | ||||
|   save-created-theme: "保存主題" | ||||
| common/views/components/signin.vue: | ||||
|   signin-with-twitter: "用 Twitter 帳號登入" | ||||
|   signin-with-github: "用 GitHub 帳號登入" | ||||
|   signin-with-discord: "用 Discord 帳號登入" | ||||
|   login-failed: "登錄失敗。 請檢查用戶名和密碼。" | ||||
| common/views/components/signup.vue: | ||||
|   invitation-code: "邀請碼" | ||||
|   username: "用戶名" | ||||
|   available: "可用" | ||||
|   too-long: "請不要超過20個字元" | ||||
|   password: "密碼" | ||||
|   password-placeholder: "建議至少8個字元" | ||||
| common/views/components/stream-indicator.vue: | ||||
|   connecting: "正在連線" | ||||
|   reconnecting: "正在重新連線" | ||||
|   connected: "已建立連線" | ||||
| common/views/components/integration-settings.vue: | ||||
|   disconnect: "中斷連線" | ||||
| common/views/components/github-setting.vue: | ||||
|   reconnect: "重新連線" | ||||
|   disconnect: "中斷連線" | ||||
| common/views/components/discord-setting.vue: | ||||
|   reconnect: "重新連線" | ||||
|   disconnect: "中斷連線" | ||||
| common/views/components/language-settings.vue: | ||||
|   recommended: "推薦" | ||||
|   auto: "自動" | ||||
|   specify-language: "指定語言" | ||||
| common/views/components/profile-editor.vue: | ||||
|   title: "個人資料" | ||||
|   name: "名稱" | ||||
|   birthday: "生日:" | ||||
|   privacy: "隱私" | ||||
| admin/views/dashboard.vue: | ||||
|   drive: "雲端硬碟" | ||||
| admin/views/charts.vue: | ||||
|   drive: "雲端硬碟" | ||||
| pages: | ||||
|   like: "贊" | ||||
							
								
								
									
										31
									
								
								migration/1556348509290-Pages.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								migration/1556348509290-Pages.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class Pages1556348509290 implements MigrationInterface { | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`CREATE TYPE "page_visibility_enum" AS ENUM('public', 'followers', 'specified')`); | ||||
|         await queryRunner.query(`CREATE TABLE "page" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "updatedAt" TIMESTAMP WITH TIME ZONE NOT NULL, "title" character varying(256) NOT NULL, "name" character varying(256) NOT NULL, "summary" character varying(256), "alignCenter" boolean NOT NULL, "font" character varying(32) NOT NULL, "userId" character varying(32) NOT NULL, "eyeCatchingImageId" character varying(32), "content" jsonb NOT NULL DEFAULT '[]', "variables" jsonb NOT NULL DEFAULT '[]', "visibility" "page_visibility_enum" NOT NULL, "visibleUserIds" character varying(32) array NOT NULL DEFAULT '{}'::varchar[], CONSTRAINT "PK_742f4117e065c5b6ad21b37ba1f" PRIMARY KEY ("id"))`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_fbb4297c927a9b85e9cefa2eb1" ON "page" ("createdAt") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_af639b066dfbca78b01a920f8a" ON "page" ("updatedAt") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_b82c19c08afb292de4600d99e4" ON "page" ("name") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_ae1d917992dd0c9d9bbdad06c4" ON "page" ("userId") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_90148bbc2bf0854428786bfc15" ON "page" ("visibleUserIds") `); | ||||
|         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_2133ef8317e4bdb839c0dcbf13" ON "page" ("userId", "name") `); | ||||
|         await queryRunner.query(`ALTER TABLE "page" ADD CONSTRAINT "FK_ae1d917992dd0c9d9bbdad06c4a" FOREIGN KEY ("userId") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "page" ADD CONSTRAINT "FK_3126dd7c502c9e4d7597ef7ef10" FOREIGN KEY ("eyeCatchingImageId") REFERENCES "drive_file"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "page" DROP CONSTRAINT "FK_3126dd7c502c9e4d7597ef7ef10"`); | ||||
|         await queryRunner.query(`ALTER TABLE "page" DROP CONSTRAINT "FK_ae1d917992dd0c9d9bbdad06c4a"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_2133ef8317e4bdb839c0dcbf13"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_90148bbc2bf0854428786bfc15"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_ae1d917992dd0c9d9bbdad06c4"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_b82c19c08afb292de4600d99e4"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_af639b066dfbca78b01a920f8a"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_fbb4297c927a9b85e9cefa2eb1"`); | ||||
|         await queryRunner.query(`DROP TABLE "page"`); | ||||
|         await queryRunner.query(`DROP TYPE "page_visibility_enum"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										23
									
								
								migration/1556746559567-UserProfile.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								migration/1556746559567-UserProfile.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class UserProfile1556746559567 implements MigrationInterface { | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`UPDATE "user_profile" SET github = FALSE`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "githubId"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD COLUMN "githubId" VARCHAR(64)`); | ||||
|         await queryRunner.query(`UPDATE "user_profile" SET discord = FALSE`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "discordExpiresDate"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD COLUMN "discordExpiresDate" VARCHAR(64)`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`UPDATE "user_profile" SET github = FALSE`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "githubId"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD COLUMN "githubId" INTEGER`); | ||||
|         await queryRunner.query(`UPDATE "user_profile" SET discord = FALSE`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "discordExpiresDate"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD COLUMN "discordExpiresDate" INTEGER`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										13
									
								
								migration/1557476068003-PinnedUsers.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								migration/1557476068003-PinnedUsers.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class PinnedUsers1557476068003 implements MigrationInterface { | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "pinnedUsers" character varying(256) array NOT NULL DEFAULT '{}'::varchar[]`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "pinnedUsers"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										16
									
								
								migration/1557761316509-AddSomeUrls.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								migration/1557761316509-AddSomeUrls.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class AddSomeUrls1557761316509 implements MigrationInterface { | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "ToSUrl" character varying(512)`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "repositoryUrl" character varying(512) NOT NULL DEFAULT 'https://github.com/syuilo/misskey'`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "feedbackUrl" character varying(512) DEFAULT 'https://github.com/syuilo/misskey/issues/new'`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "feedbackUrl"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "repositoryUrl"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "ToSUrl"`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										31
									
								
								migration/1557932705754-ObjectStorageSetting.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								migration/1557932705754-ObjectStorageSetting.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class ObjectStorageSetting1557932705754 implements MigrationInterface { | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "useObjectStorage" boolean NOT NULL DEFAULT false`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStorageBucket" character varying(512)`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStoragePrefix" character varying(512)`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStorageBaseUrl" character varying(512)`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStorageEndpoint" character varying(512)`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStorageRegion" character varying(512)`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStorageAccessKey" character varying(512)`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStorageSecretKey" character varying(512)`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStoragePort" integer`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStorageUseSSL" boolean NOT NULL DEFAULT true`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStorageUseSSL"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStoragePort"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStorageSecretKey"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStorageAccessKey"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStorageRegion"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStorageEndpoint"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStorageBaseUrl"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStoragePrefix"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStorageBucket"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "useObjectStorage"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										23
									
								
								migration/1558072954435-PageLike.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								migration/1558072954435-PageLike.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class PageLike1558072954435 implements MigrationInterface { | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`CREATE TABLE "page_like" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "userId" character varying(32) NOT NULL, "pageId" character varying(32) NOT NULL, CONSTRAINT "PK_813f034843af992d3ae0f43c64c" PRIMARY KEY ("id"))`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_0e61efab7f88dbb79c9166dbb4" ON "page_like" ("userId") `); | ||||
|         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_4ce6fb9c70529b4c8ac46c9bfa" ON "page_like" ("userId", "pageId") `); | ||||
|         await queryRunner.query(`ALTER TABLE "page" ADD "likedCount" integer NOT NULL DEFAULT 0`); | ||||
|         await queryRunner.query(`ALTER TABLE "page_like" ADD CONSTRAINT "FK_0e61efab7f88dbb79c9166dbb48" FOREIGN KEY ("userId") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "page_like" ADD CONSTRAINT "FK_cf8782626dced3176038176a847" FOREIGN KEY ("pageId") REFERENCES "page"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "page_like" DROP CONSTRAINT "FK_cf8782626dced3176038176a847"`); | ||||
|         await queryRunner.query(`ALTER TABLE "page_like" DROP CONSTRAINT "FK_0e61efab7f88dbb79c9166dbb48"`); | ||||
|         await queryRunner.query(`ALTER TABLE "page" DROP COLUMN "likedCount"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_4ce6fb9c70529b4c8ac46c9bfa"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_0e61efab7f88dbb79c9166dbb4"`); | ||||
|         await queryRunner.query(`DROP TABLE "page_like"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										41
									
								
								migration/1558103093633-UserGroup.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								migration/1558103093633-UserGroup.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class UserGroup1558103093633 implements MigrationInterface { | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`CREATE TABLE "user_group" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "name" character varying(256) NOT NULL, "userId" character varying(32) NOT NULL, "isPrivate" boolean NOT NULL DEFAULT false, CONSTRAINT "PK_3c29fba6fe013ec8724378ce7c9" PRIMARY KEY ("id"))`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_20e30aa35180e317e133d75316" ON "user_group" ("createdAt") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_3d6b372788ab01be58853003c9" ON "user_group" ("userId") `); | ||||
|         await queryRunner.query(`CREATE TABLE "user_group_joining" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "userId" character varying(32) NOT NULL, "userGroupId" character varying(32) NOT NULL, CONSTRAINT "PK_15f2425885253c5507e1599cfe7" PRIMARY KEY ("id"))`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_f3a1b4bd0c7cabba958a0c0b23" ON "user_group_joining" ("userId") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_67dc758bc0566985d1b3d39986" ON "user_group_joining" ("userGroupId") `); | ||||
|         await queryRunner.query(`ALTER TABLE "messaging_message" ADD "groupId" character varying(32)`); | ||||
|         await queryRunner.query(`ALTER TABLE "messaging_message" ADD "reads" character varying(32) array NOT NULL DEFAULT '{}'::varchar[]`); | ||||
|         await queryRunner.query(`ALTER TABLE "messaging_message" ALTER COLUMN "recipientId" DROP NOT NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "messaging_message"."recipientId" IS 'The recipient user ID.'`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_2c4be03b446884f9e9c502135b" ON "messaging_message" ("groupId") `); | ||||
|         await queryRunner.query(`ALTER TABLE "messaging_message" ADD CONSTRAINT "FK_2c4be03b446884f9e9c502135be" FOREIGN KEY ("groupId") REFERENCES "user_group"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_group" ADD CONSTRAINT "FK_3d6b372788ab01be58853003c93" FOREIGN KEY ("userId") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_group_joining" ADD CONSTRAINT "FK_f3a1b4bd0c7cabba958a0c0b231" FOREIGN KEY ("userId") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_group_joining" ADD CONSTRAINT "FK_67dc758bc0566985d1b3d399865" FOREIGN KEY ("userGroupId") REFERENCES "user_group"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "user_group_joining" DROP CONSTRAINT "FK_67dc758bc0566985d1b3d399865"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_group_joining" DROP CONSTRAINT "FK_f3a1b4bd0c7cabba958a0c0b231"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_group" DROP CONSTRAINT "FK_3d6b372788ab01be58853003c93"`); | ||||
|         await queryRunner.query(`ALTER TABLE "messaging_message" DROP CONSTRAINT "FK_2c4be03b446884f9e9c502135be"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_2c4be03b446884f9e9c502135b"`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "messaging_message"."recipientId" IS ''`); | ||||
|         await queryRunner.query(`ALTER TABLE "messaging_message" ALTER COLUMN "recipientId" SET NOT NULL`); | ||||
|         await queryRunner.query(`ALTER TABLE "messaging_message" DROP COLUMN "reads"`); | ||||
|         await queryRunner.query(`ALTER TABLE "messaging_message" DROP COLUMN "groupId"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_67dc758bc0566985d1b3d39986"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_f3a1b4bd0c7cabba958a0c0b23"`); | ||||
|         await queryRunner.query(`DROP TABLE "user_group_joining"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_3d6b372788ab01be58853003c9"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_20e30aa35180e317e133d75316"`); | ||||
|         await queryRunner.query(`DROP TABLE "user_group"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										25
									
								
								migration/1558257926829-UserGroupInvite.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								migration/1558257926829-UserGroupInvite.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class UserGroupInvite1558257926829 implements MigrationInterface { | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`CREATE TABLE "user_group_invite" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "userId" character varying(32) NOT NULL, "userGroupId" character varying(32) NOT NULL, CONSTRAINT "PK_3893884af0d3a5f4d01e7921a97" PRIMARY KEY ("id"))`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_1039988afa3bf991185b277fe0" ON "user_group_invite" ("userId") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_e10924607d058004304611a436" ON "user_group_invite" ("userGroupId") `); | ||||
|         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_78787741f9010886796f2320a4" ON "user_group_invite" ("userId", "userGroupId") `); | ||||
|         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_d9ecaed8c6dc43f3592c229282" ON "user_group_joining" ("userId", "userGroupId") `); | ||||
|         await queryRunner.query(`ALTER TABLE "user_group_invite" ADD CONSTRAINT "FK_1039988afa3bf991185b277fe03" FOREIGN KEY ("userId") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_group_invite" ADD CONSTRAINT "FK_e10924607d058004304611a436a" FOREIGN KEY ("userGroupId") REFERENCES "user_group"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`ALTER TABLE "user_group_invite" DROP CONSTRAINT "FK_e10924607d058004304611a436a"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_group_invite" DROP CONSTRAINT "FK_1039988afa3bf991185b277fe03"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_d9ecaed8c6dc43f3592c229282"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_78787741f9010886796f2320a4"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_e10924607d058004304611a436"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_1039988afa3bf991185b277fe0"`); | ||||
|         await queryRunner.query(`DROP TABLE "user_group_invite"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										13
									
								
								migration/1558266512381-UserListJoining.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								migration/1558266512381-UserListJoining.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class UserListJoining1558266512381 implements MigrationInterface { | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_90f7da835e4c10aca6853621e1" ON "user_list_joining" ("userId", "userListId") `); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<any> { | ||||
|         await queryRunner.query(`DROP INDEX "IDX_90f7da835e4c10aca6853621e1"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										204
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										204
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,9 +1,8 @@ | ||||
| { | ||||
| 	"name": "misskey", | ||||
| 	"author": "syuilo <i@syuilo.com>", | ||||
| 	"version": "10.86.0", | ||||
| 	"clientVersion": "2.0.14319", | ||||
| 	"codename": "nighthike", | ||||
| 	"version": "11.18.0", | ||||
| 	"codename": "daybreak", | ||||
| 	"repository": { | ||||
| 		"type": "git", | ||||
| 		"url": "https://github.com/syuilo/misskey.git" | ||||
| @@ -12,7 +11,10 @@ | ||||
| 	"private": true, | ||||
| 	"scripts": { | ||||
| 		"start": "node ./index.js", | ||||
| 		"debug": "DEBUG=misskey:* node ./index.js", | ||||
| 		"init": "node ./built/init.js", | ||||
| 		"ormconfig": "node ./built/ormconfig.js", | ||||
| 		"migrate": "npm run ormconfig && ts-node ./node_modules/typeorm/cli.js migration:run", | ||||
| 		"migrateandstart": "npm run migrate && npm run start", | ||||
| 		"build": "webpack && gulp build", | ||||
| 		"webpack": "webpack", | ||||
| 		"watch": "webpack --watch", | ||||
| @@ -24,131 +26,127 @@ | ||||
| 		"format": "gulp format" | ||||
| 	}, | ||||
| 	"dependencies": { | ||||
| 		"@fortawesome/fontawesome-svg-core": "1.2.14", | ||||
| 		"@fortawesome/free-brands-svg-icons": "5.7.1", | ||||
| 		"@fortawesome/free-regular-svg-icons": "5.7.0", | ||||
| 		"@fortawesome/free-solid-svg-icons": "5.7.1", | ||||
| 		"@elastic/elasticsearch": "7.0.0-rc.2", | ||||
| 		"@fortawesome/fontawesome-svg-core": "1.2.15", | ||||
| 		"@fortawesome/free-brands-svg-icons": "5.7.2", | ||||
| 		"@fortawesome/free-regular-svg-icons": "5.7.2", | ||||
| 		"@fortawesome/free-solid-svg-icons": "5.7.2", | ||||
| 		"@fortawesome/vue-fontawesome": "0.1.5", | ||||
| 		"@koa/cors": "2.2.3", | ||||
| 		"@prezzemolo/rap": "0.1.2", | ||||
| 		"@prezzemolo/zip": "0.0.3", | ||||
| 		"@types/bcryptjs": "2.4.2", | ||||
| 		"@types/chai-http": "3.0.5", | ||||
| 		"@types/bull": "3.5.11", | ||||
| 		"@types/dateformat": "3.0.0", | ||||
| 		"@types/deep-equal": "1.0.1", | ||||
| 		"@types/double-ended-queue": "2.1.0", | ||||
| 		"@types/elasticsearch": "5.0.30", | ||||
| 		"@types/file-type": "10.6.0", | ||||
| 		"@types/gulp": "4.0.5", | ||||
| 		"@types/file-type": "10.9.1", | ||||
| 		"@types/gulp": "4.0.6", | ||||
| 		"@types/gulp-mocha": "0.0.32", | ||||
| 		"@types/gulp-rename": "0.0.33", | ||||
| 		"@types/gulp-replace": "0.0.31", | ||||
| 		"@types/gulp-uglify": "3.0.6", | ||||
| 		"@types/gulp-util": "3.0.34", | ||||
| 		"@types/is-root": "1.0.0", | ||||
| 		"@types/is-svg": "3.0.0", | ||||
| 		"@types/is-url": "1.2.28", | ||||
| 		"@types/js-yaml": "3.12.0", | ||||
| 		"@types/jsdom": "12.2.1", | ||||
| 		"@types/katex": "0.5.0", | ||||
| 		"@types/js-yaml": "3.12.1", | ||||
| 		"@types/jsdom": "12.2.3", | ||||
| 		"@types/katex": "0.10.1", | ||||
| 		"@types/koa": "2.0.48", | ||||
| 		"@types/koa-bodyparser": "5.0.2", | ||||
| 		"@types/koa-compress": "2.0.8", | ||||
| 		"@types/koa-compress": "2.0.9", | ||||
| 		"@types/koa-cors": "0.0.0", | ||||
| 		"@types/koa-favicon": "2.0.19", | ||||
| 		"@types/koa-logger": "3.1.1", | ||||
| 		"@types/koa-mount": "3.0.1", | ||||
| 		"@types/koa-multer": "1.0.0", | ||||
| 		"@types/koa-router": "7.0.39", | ||||
| 		"@types/koa-send": "4.1.1", | ||||
| 		"@types/koa-router": "7.0.40", | ||||
| 		"@types/koa-send": "4.1.2", | ||||
| 		"@types/koa-views": "2.0.3", | ||||
| 		"@types/koa__cors": "2.2.3", | ||||
| 		"@types/lolex": "3.1.1", | ||||
| 		"@types/minio": "7.0.1", | ||||
| 		"@types/mkdirp": "0.5.2", | ||||
| 		"@types/mocha": "5.2.5", | ||||
| 		"@types/mongodb": "3.1.19", | ||||
| 		"@types/node": "10.12.21", | ||||
| 		"@types/nodemailer": "4.6.5", | ||||
| 		"@types/mocha": "5.2.6", | ||||
| 		"@types/node": "11.13.8", | ||||
| 		"@types/nodemailer": "4.6.8", | ||||
| 		"@types/nprogress": "0.0.29", | ||||
| 		"@types/oauth": "0.9.1", | ||||
| 		"@types/parse5": "5.0.0", | ||||
| 		"@types/parsimmon": "1.10.0", | ||||
| 		"@types/portscanner": "2.1.0", | ||||
| 		"@types/pug": "2.0.4", | ||||
| 		"@types/qrcode": "1.3.0", | ||||
| 		"@types/qrcode": "1.3.2", | ||||
| 		"@types/random-seed": "0.3.3", | ||||
| 		"@types/ratelimiter": "2.1.28", | ||||
| 		"@types/redis": "2.8.10", | ||||
| 		"@types/redis": "2.8.12", | ||||
| 		"@types/rename": "1.0.1", | ||||
| 		"@types/request": "2.48.1", | ||||
| 		"@types/request-promise-native": "1.0.15", | ||||
| 		"@types/request-promise-native": "1.0.16", | ||||
| 		"@types/request-stats": "3.0.0", | ||||
| 		"@types/rimraf": "2.0.2", | ||||
| 		"@types/seedrandom": "2.4.27", | ||||
| 		"@types/sharp": "0.21.1", | ||||
| 		"@types/seedrandom": "2.4.28", | ||||
| 		"@types/sharp": "0.22.1", | ||||
| 		"@types/showdown": "1.9.2", | ||||
| 		"@types/speakeasy": "2.0.3", | ||||
| 		"@types/speakeasy": "2.0.4", | ||||
| 		"@types/systeminformation": "3.23.1", | ||||
| 		"@types/tinycolor2": "1.4.1", | ||||
| 		"@types/tmp": "0.0.33", | ||||
| 		"@types/tmp": "0.1.0", | ||||
| 		"@types/uuid": "3.4.4", | ||||
| 		"@types/web-push": "3.3.0", | ||||
| 		"@types/webpack": "4.4.24", | ||||
| 		"@types/webpack": "4.4.29", | ||||
| 		"@types/webpack-stream": "3.2.10", | ||||
| 		"@types/websocket": "0.0.40", | ||||
| 		"@types/ws": "6.0.1", | ||||
| 		"animejs": "3.0.1", | ||||
| 		"apexcharts": "3.3.0", | ||||
| 		"apexcharts": "3.6.12", | ||||
| 		"autobind-decorator": "2.4.0", | ||||
| 		"autosize": "4.0.2", | ||||
| 		"autwh": "0.1.0", | ||||
| 		"bcryptjs": "2.4.3", | ||||
| 		"bee-queue": "1.2.2", | ||||
| 		"bootstrap-vue": "2.0.0-rc.11", | ||||
| 		"cafy": "14.0.1", | ||||
| 		"bootstrap-vue": "2.0.0-rc.13", | ||||
| 		"bull": "3.9.1", | ||||
| 		"cafy": "15.1.1", | ||||
| 		"chai": "4.2.0", | ||||
| 		"chai-http": "4.2.1", | ||||
| 		"chalk": "2.4.2", | ||||
| 		"commander": "2.19.0", | ||||
| 		"cli-highlight": "2.1.1", | ||||
| 		"commander": "2.20.0", | ||||
| 		"content-disposition": "0.5.3", | ||||
| 		"crc-32": "1.2.0", | ||||
| 		"css-loader": "2.1.0", | ||||
| 		"cssnano": "4.1.8", | ||||
| 		"css-loader": "2.1.1", | ||||
| 		"cssnano": "4.1.10", | ||||
| 		"dateformat": "3.0.3", | ||||
| 		"deep-equal": "1.0.1", | ||||
| 		"deepcopy": "0.6.3", | ||||
| 		"diskusage": "1.0.0", | ||||
| 		"diskusage": "1.1.1", | ||||
| 		"double-ended-queue": "2.1.0-0", | ||||
| 		"elasticsearch": "15.3.1", | ||||
| 		"emojilib": "2.4.0", | ||||
| 		"escape-regexp": "0.0.1", | ||||
| 		"eslint": "5.12.0", | ||||
| 		"eslint-plugin-vue": "5.1.0", | ||||
| 		"eventemitter3": "3.1.0", | ||||
| 		"feed": "2.0.2", | ||||
| 		"file-type": "10.7.1", | ||||
| 		"eslint": "5.16.0", | ||||
| 		"eslint-plugin-vue": "5.2.2", | ||||
| 		"eventemitter3": "3.1.2", | ||||
| 		"feed": "2.0.4", | ||||
| 		"file-type": "10.11.0", | ||||
| 		"fuckadblock": "3.2.1", | ||||
| 		"gulp": "4.0.0", | ||||
| 		"gulp": "4.0.2", | ||||
| 		"gulp-cssnano": "2.1.3", | ||||
| 		"gulp-imagemin": "5.0.3", | ||||
| 		"gulp-mocha": "6.0.0", | ||||
| 		"gulp-rename": "1.4.0", | ||||
| 		"gulp-replace": "1.0.0", | ||||
| 		"gulp-sourcemaps": "2.6.4", | ||||
| 		"gulp-sourcemaps": "2.6.5", | ||||
| 		"gulp-stylus": "2.7.0", | ||||
| 		"gulp-tslint": "8.1.3", | ||||
| 		"gulp-typescript": "5.0.0", | ||||
| 		"gulp-uglify": "3.0.1", | ||||
| 		"gulp-tslint": "8.1.4", | ||||
| 		"gulp-typescript": "5.0.1", | ||||
| 		"gulp-uglify": "3.0.2", | ||||
| 		"gulp-util": "3.0.8", | ||||
| 		"gulp-yaml": "2.0.3", | ||||
| 		"hard-source-webpack-plugin": "0.13.1", | ||||
| 		"html-minifier": "3.5.21", | ||||
| 		"html-minifier": "4.0.0", | ||||
| 		"http-signature": "1.2.0", | ||||
| 		"insert-text-at-cursor": "0.1.2", | ||||
| 		"is-root": "2.0.0", | ||||
| 		"is-svg": "3.0.0", | ||||
| 		"js-yaml": "3.12.1", | ||||
| 		"jsdom": "13.2.0", | ||||
| 		"insert-text-at-cursor": "0.2.0", | ||||
| 		"is-root": "2.1.0", | ||||
| 		"is-svg": "4.1.0", | ||||
| 		"js-yaml": "3.13.1", | ||||
| 		"jsdom": "15.1.0", | ||||
| 		"json5": "2.1.0", | ||||
| 		"json5-loader": "1.0.1", | ||||
| 		"katex": "0.10.0", | ||||
| 		"json5-loader": "2.0.0", | ||||
| 		"katex": "0.10.1", | ||||
| 		"koa": "2.7.0", | ||||
| 		"koa-bodyparser": "4.2.1", | ||||
| 		"koa-compress": "3.0.0", | ||||
| @@ -160,50 +158,53 @@ | ||||
| 		"koa-router": "7.4.0", | ||||
| 		"koa-send": "5.0.0", | ||||
| 		"koa-slow": "2.1.0", | ||||
| 		"koa-views": "6.1.5", | ||||
| 		"koa-views": "6.2.0", | ||||
| 		"langmap": "0.0.16", | ||||
| 		"loader-utils": "1.2.3", | ||||
| 		"lolex": "3.1.0", | ||||
| 		"lookup-dns-cache": "2.1.0", | ||||
| 		"minio": "7.0.5", | ||||
| 		"mkdirp": "0.5.1", | ||||
| 		"mocha": "5.2.0", | ||||
| 		"minio": "7.0.8", | ||||
| 		"mocha": "6.1.4", | ||||
| 		"moji": "0.5.1", | ||||
| 		"moment": "2.24.0", | ||||
| 		"mongodb": "3.1.13", | ||||
| 		"monk": "6.0.6", | ||||
| 		"ms": "2.1.1", | ||||
| 		"nan": "2.12.1", | ||||
| 		"nested-property": "0.0.7", | ||||
| 		"nodemailer": "5.1.1", | ||||
| 		"node-fetch": "2.5.0", | ||||
| 		"nodemailer": "6.1.1", | ||||
| 		"nprogress": "0.2.0", | ||||
| 		"object-assign-deep": "0.4.0", | ||||
| 		"on-build-webpack": "0.1.0", | ||||
| 		"os-utils": "0.0.14", | ||||
| 		"parse5": "5.1.0", | ||||
| 		"parsimmon": "1.12.0", | ||||
| 		"pg": "7.11.0", | ||||
| 		"portscanner": "2.2.0", | ||||
| 		"postcss-loader": "3.0.0", | ||||
| 		"prismjs": "1.15.0", | ||||
| 		"prismjs": "1.16.0", | ||||
| 		"progress-bar-webpack-plugin": "1.12.1", | ||||
| 		"promise-any": "0.2.0", | ||||
| 		"promise-limit": "2.7.0", | ||||
| 		"promise-sequential": "1.1.1", | ||||
| 		"pug": "2.0.3", | ||||
| 		"punycode": "2.1.1", | ||||
| 		"pureimage": "0.1.6", | ||||
| 		"qrcode": "1.3.3", | ||||
| 		"randomcolor": "0.5.3", | ||||
| 		"ratelimiter": "3.2.0", | ||||
| 		"random-seed": "0.3.0", | ||||
| 		"randomcolor": "0.5.4", | ||||
| 		"ratelimiter": "3.3.0", | ||||
| 		"recaptcha-promise": "0.1.3", | ||||
| 		"reconnecting-websocket": "4.1.10", | ||||
| 		"redis": "2.8.0", | ||||
| 		"reflect-metadata": "0.1.13", | ||||
| 		"rename": "1.0.4", | ||||
| 		"request": "2.88.0", | ||||
| 		"request-promise-native": "1.0.5", | ||||
| 		"request-promise-native": "1.0.7", | ||||
| 		"request-stats": "3.0.0", | ||||
| 		"require-all": "3.0.0", | ||||
| 		"rimraf": "2.6.3", | ||||
| 		"rndstr": "1.0.0", | ||||
| 		"s-age": "1.1.2", | ||||
| 		"seedrandom": "2.4.4", | ||||
| 		"sharp": "0.21.3", | ||||
| 		"seedrandom": "3.0.1", | ||||
| 		"sharp": "0.22.1", | ||||
| 		"showdown": "1.9.0", | ||||
| 		"showdown-highlightjs-extension": "0.1.2", | ||||
| 		"speakeasy": "2.0.0", | ||||
| @@ -212,47 +213,50 @@ | ||||
| 		"stylus": "0.54.5", | ||||
| 		"stylus-loader": "3.0.2", | ||||
| 		"summaly": "2.2.0", | ||||
| 		"systeminformation": "3.54.0", | ||||
| 		"systeminformation": "4.1.6", | ||||
| 		"syuilo-password-strength": "0.0.1", | ||||
| 		"terser-webpack-plugin": "1.2.2", | ||||
| 		"terser-webpack-plugin": "1.2.4", | ||||
| 		"textarea-caret": "3.1.0", | ||||
| 		"tinycolor2": "1.4.1", | ||||
| 		"tmp": "0.0.33", | ||||
| 		"tmp": "0.1.0", | ||||
| 		"ts-loader": "5.3.3", | ||||
| 		"ts-node": "7.0.1", | ||||
| 		"tslint": "5.12.1", | ||||
| 		"tslint": "5.15.0", | ||||
| 		"tslint-sonarts": "1.9.0", | ||||
| 		"typescript": "3.2.4", | ||||
| 		"typescript-eslint-parser": "21.0.2", | ||||
| 		"typeorm": "0.2.16-rc.1", | ||||
| 		"typescript": "3.3.3333", | ||||
| 		"typescript-eslint-parser": "22.0.0", | ||||
| 		"uglify-es": "3.3.9", | ||||
| 		"ulid": "2.3.0", | ||||
| 		"url-loader": "1.1.2", | ||||
| 		"uuid": "3.3.2", | ||||
| 		"v-animate-css": "0.0.3", | ||||
| 		"v-debounce": "0.1.2", | ||||
| 		"video-thumbnail-generator": "1.1.3", | ||||
| 		"vue": "2.6.6", | ||||
| 		"vue": "2.6.10", | ||||
| 		"vue-color": "2.7.0", | ||||
| 		"vue-content-loading": "1.5.3", | ||||
| 		"vue-content-loading": "1.6.0", | ||||
| 		"vue-cropperjs": "3.0.0", | ||||
| 		"vue-i18n": "8.8.1", | ||||
| 		"vue-js-modal": "1.3.28", | ||||
| 		"vue-loader": "15.6.2", | ||||
| 		"vue-i18n": "8.11.2", | ||||
| 		"vue-js-modal": "1.3.31", | ||||
| 		"vue-json-pretty": "1.6.0", | ||||
| 		"vue-loader": "15.7.0", | ||||
| 		"vue-marquee-text-component": "1.1.1", | ||||
| 		"vue-prism-component": "1.1.1", | ||||
| 		"vue-router": "3.0.2", | ||||
| 		"vue-router": "3.0.6", | ||||
| 		"vue-sequential-entrance": "1.1.3", | ||||
| 		"vue-style-loader": "4.1.2", | ||||
| 		"vue-svg-inline-loader": "1.2.10", | ||||
| 		"vue-template-compiler": "2.6.6", | ||||
| 		"vuedraggable": "2.17.0", | ||||
| 		"vue-svg-inline-loader": "1.2.15", | ||||
| 		"vue-template-compiler": "2.6.10", | ||||
| 		"vuedraggable": "2.21.0", | ||||
| 		"vuewordcloud": "18.7.11", | ||||
| 		"vuex": "3.1.0", | ||||
| 		"vuex": "3.1.1", | ||||
| 		"vuex-persistedstate": "2.5.4", | ||||
| 		"web-push": "3.3.3", | ||||
| 		"webfinger.js": "2.7.0", | ||||
| 		"webpack": "4.28.4", | ||||
| 		"webpack-cli": "3.2.1", | ||||
| 		"web-push": "3.3.5", | ||||
| 		"webpack": "4.32.0", | ||||
| 		"webpack-cli": "3.3.2", | ||||
| 		"websocket": "1.0.28", | ||||
| 		"ws": "6.1.3", | ||||
| 		"ws": "7.0.0", | ||||
| 		"xev": "2.0.1" | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										17
									
								
								src/@types/deepcopy.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										17
									
								
								src/@types/deepcopy.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -1,17 +0,0 @@ | ||||
| declare module 'deepcopy'; | ||||
|  | ||||
| declare namespace deepcopy { | ||||
| 	type DeepcopyCustomizerValueType = 'Object'; | ||||
|  | ||||
| 	type DeepcopyCustomizer<T> = ( | ||||
| 		value: T, | ||||
| 		valueType: DeepcopyCustomizerValueType) => T; | ||||
|  | ||||
| 	interface DeepcopyOptions<T> { | ||||
| 		customizer: DeepcopyCustomizer<T>; | ||||
| 	} | ||||
|  | ||||
| 	export function deepcopy<T>( | ||||
| 		value: T, | ||||
| 		options?: DeepcopyOptions<T> | DeepcopyCustomizer<T>): T; | ||||
| } | ||||
							
								
								
									
										7
									
								
								src/@types/escape-regexp.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								src/@types/escape-regexp.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +0,0 @@ | ||||
| declare module 'escape-regexp' { | ||||
| 	function escapeRegExp(str: string): string; | ||||
|  | ||||
| 	namespace escapeRegExp {} // Hack | ||||
|  | ||||
| 	export = escapeRegExp; | ||||
| } | ||||
							
								
								
									
										2
									
								
								src/@types/koa-slow.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								src/@types/koa-slow.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -8,7 +8,7 @@ declare module 'koa-slow' { | ||||
|  | ||||
| 	function slow(options?: ISlowOptions): Middleware; | ||||
|  | ||||
| 	namespace slow { } // Hack | ||||
| 	namespace slow {} // Hack | ||||
|  | ||||
| 	export = slow; | ||||
| } | ||||
|   | ||||
							
								
								
									
										65
									
								
								src/@types/webfinger.js.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										65
									
								
								src/@types/webfinger.js.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -1,65 +0,0 @@ | ||||
| declare module 'webfinger.js' { | ||||
| 	interface IWebFingerConstructorConfig { | ||||
| 		tls_only?: boolean; | ||||
| 		webfist_fallback?: boolean; | ||||
| 		uri_fallback?: boolean; | ||||
| 		request_timeout?: number; | ||||
| 	} | ||||
|  | ||||
| 	type JRDProperties = { [type: string]: string }; | ||||
|  | ||||
| 	interface IJRDLink { | ||||
| 		rel: string; | ||||
| 		type?: string; | ||||
| 		href?: string; | ||||
| 		template?: string; | ||||
| 		titles?: { [lang: string]: string }; | ||||
| 		properties?: JRDProperties; | ||||
| 	} | ||||
|  | ||||
| 	interface IJRD { | ||||
| 		subject?: string; | ||||
| 		expires?: Date; | ||||
| 		aliases?: string[]; | ||||
| 		properties?: JRDProperties; | ||||
| 		links?: IJRDLink[]; | ||||
| 	} | ||||
|  | ||||
| 	interface IIDXLinks { | ||||
|     'avatar': IJRDLink[]; | ||||
|     'remotestorage': IJRDLink[]; | ||||
|     'blog': IJRDLink[]; | ||||
|     'vcard': IJRDLink[]; | ||||
|     'updates': IJRDLink[]; | ||||
|     'share': IJRDLink[]; | ||||
|     'profile': IJRDLink[]; | ||||
|     'webfist': IJRDLink[]; | ||||
| 		'camlistore': IJRDLink[]; | ||||
| 		[type: string]: IJRDLink[]; | ||||
| 	} | ||||
|  | ||||
| 	interface IIDXProperties { | ||||
| 		'name': string; | ||||
| 		[type: string]: string; | ||||
| 	} | ||||
|  | ||||
| 	interface IIDX { | ||||
| 		links: IIDXLinks; | ||||
| 		properties: IIDXProperties; | ||||
| 	} | ||||
|  | ||||
| 	interface ILookupCallbackResult { | ||||
| 		object: IJRD; | ||||
| 		json: string; | ||||
| 		idx: IIDX; | ||||
| 	} | ||||
|  | ||||
| 	type LookupCallback = (err: Error | string, result?: ILookupCallbackResult) => void; | ||||
|  | ||||
| 	export class WebFinger { | ||||
| 		constructor(config?: IWebFingerConstructorConfig); | ||||
|  | ||||
| 		public lookup(address: string, cb: LookupCallback): NodeJS.Timeout; | ||||
| 		public lookupLink(address: string, rel: string, cb: IJRDLink): void; | ||||
| 	} | ||||
| } | ||||
| @@ -5,8 +5,7 @@ program | ||||
| 	.version(pkg.version) | ||||
| 	.option('--no-daemons', 'Disable daemon processes (for debbuging)') | ||||
| 	.option('--disable-clustering', 'Disable clustering') | ||||
| 	.option('--disable-queue', 'Disable job queue processing') | ||||
| 	.option('--only-server', 'Run server only (without job queue)') | ||||
| 	.option('--only-server', 'Run server only (without job queue processing)') | ||||
| 	.option('--only-queue', 'Pocessing job queue only (without server)') | ||||
| 	.option('--quiet', 'Suppress all logs') | ||||
| 	.option('--verbose', 'Enable all logs') | ||||
| @@ -15,7 +14,9 @@ program | ||||
| 	.option('--color', 'This option is a dummy for some external program\'s (e.g. forever) issue.') | ||||
| 	.parse(process.argv); | ||||
|  | ||||
| /*if (process.env.MK_DISABLE_QUEUE)*/ program.disableQueue = true; | ||||
| if (process.env.MK_ONLY_QUEUE) program.onlyQueue = true; | ||||
| if (process.env.NODE_ENV === 'test') program.disableClustering = true; | ||||
| if (process.env.NODE_ENV === 'test') program.quiet = true; | ||||
| if (process.env.NODE_ENV === 'test') program.noDaemons = true; | ||||
|  | ||||
| export { program }; | ||||
|   | ||||
							
								
								
									
										77
									
								
								src/boot/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								src/boot/index.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | ||||
| import * as cluster from 'cluster'; | ||||
| import chalk from 'chalk'; | ||||
| import Xev from 'xev'; | ||||
|  | ||||
| import Logger from '../services/logger'; | ||||
| import { program } from '../argv'; | ||||
|  | ||||
| // for typeorm | ||||
| import 'reflect-metadata'; | ||||
| import { masterMain } from './master'; | ||||
| import { workerMain } from './worker'; | ||||
|  | ||||
| const logger = new Logger('core', 'cyan'); | ||||
| const clusterLogger = logger.createSubLogger('cluster', 'orange', false); | ||||
| const ev = new Xev(); | ||||
|  | ||||
| /** | ||||
|  * Init process | ||||
|  */ | ||||
| export default async function() { | ||||
| 	process.title = `Misskey (${cluster.isMaster ? 'master' : 'worker'})`; | ||||
|  | ||||
| 	if (cluster.isMaster || program.disableClustering) { | ||||
| 		await masterMain(); | ||||
|  | ||||
| 		if (cluster.isMaster) { | ||||
| 			ev.mount(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if (cluster.isWorker || program.disableClustering) { | ||||
| 		await workerMain(); | ||||
| 	} | ||||
|  | ||||
| 	// ユニットテスト時にMisskeyが子プロセスで起動された時のため | ||||
| 	// それ以外のときは process.send は使えないので弾く | ||||
| 	if (process.send) { | ||||
| 		process.send('ok'); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| //#region Events | ||||
|  | ||||
| // Listen new workers | ||||
| cluster.on('fork', worker => { | ||||
| 	clusterLogger.debug(`Process forked: [${worker.id}]`); | ||||
| }); | ||||
|  | ||||
| // Listen online workers | ||||
| cluster.on('online', worker => { | ||||
| 	clusterLogger.debug(`Process is now online: [${worker.id}]`); | ||||
| }); | ||||
|  | ||||
| // Listen for dying workers | ||||
| cluster.on('exit', worker => { | ||||
| 	// Replace the dead worker, | ||||
| 	// we're not sentimental | ||||
| 	clusterLogger.error(chalk.red(`[${worker.id}] died :(`)); | ||||
| 	cluster.fork(); | ||||
| }); | ||||
|  | ||||
| // Display detail of unhandled promise rejection | ||||
| if (!program.quiet) { | ||||
| 	process.on('unhandledRejection', console.dir); | ||||
| } | ||||
|  | ||||
| // Display detail of uncaught exception | ||||
| process.on('uncaughtException', err => { | ||||
| 	logger.error(err); | ||||
| }); | ||||
|  | ||||
| // Dying away... | ||||
| process.on('exit', code => { | ||||
| 	logger.info(`The process is going to exit with code ${code}`); | ||||
| }); | ||||
|  | ||||
| //#endregion | ||||
							
								
								
									
										176
									
								
								src/boot/master.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								src/boot/master.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | ||||
| import * as os from 'os'; | ||||
| import * as cluster from 'cluster'; | ||||
| import chalk from 'chalk'; | ||||
| import * as portscanner from 'portscanner'; | ||||
| import * as isRoot from 'is-root'; | ||||
|  | ||||
| import Logger from '../services/logger'; | ||||
| import loadConfig from '../config/load'; | ||||
| import { Config } from '../config/types'; | ||||
| import { lessThan } from '../prelude/array'; | ||||
| import * as pkg from '../../package.json'; | ||||
| import { program } from '../argv'; | ||||
| import { showMachineInfo } from '../misc/show-machine-info'; | ||||
| import { initDb } from '../db/postgre'; | ||||
|  | ||||
| const logger = new Logger('core', 'cyan'); | ||||
| const bootLogger = logger.createSubLogger('boot', 'magenta', false); | ||||
|  | ||||
| function greet() { | ||||
| 	if (!program.quiet) { | ||||
| 		//#region Misskey logo | ||||
| 		const v = `v${pkg.version}`; | ||||
| 		console.log('  _____ _         _           '); | ||||
| 		console.log(' |     |_|___ ___| |_ ___ _ _ '); | ||||
| 		console.log(' | | | | |_ -|_ -| \'_| -_| | |'); | ||||
| 		console.log(' |_|_|_|_|___|___|_,_|___|_  |'); | ||||
| 		console.log(' ' + chalk.gray(v) + ('                        |___|\n'.substr(v.length))); | ||||
| 		//#endregion | ||||
|  | ||||
| 		console.log(' Misskey is maintained by @syuilo, @AyaMorisawa, @mei23, @acid-chicken, and @rinsuki.'); | ||||
| 		console.log(chalk.keyword('orange')(' If you like Misskey, please donate to support development. https://www.patreon.com/syuilo')); | ||||
|  | ||||
| 		console.log(''); | ||||
| 		console.log(chalk`< ${os.hostname()} {gray (PID: ${process.pid.toString()})} >`); | ||||
| 	} | ||||
|  | ||||
| 	bootLogger.info('Welcome to Misskey!'); | ||||
| 	bootLogger.info(`Misskey v${pkg.version}`, null, true); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Init master process | ||||
|  */ | ||||
| export async function masterMain() { | ||||
| 	greet(); | ||||
|  | ||||
| 	let config!: Config; | ||||
|  | ||||
| 	try { | ||||
| 		// initialize app | ||||
| 		config = await init(); | ||||
|  | ||||
| 		if (config.port == null) { | ||||
| 			bootLogger.error('The port is not configured. Please configure port.', null, true); | ||||
| 			process.exit(1); | ||||
| 		} | ||||
|  | ||||
| 		if (process.platform === 'linux' && isWellKnownPort(config.port) && !isRoot()) { | ||||
| 			bootLogger.error('You need root privileges to listen on well-known port on Linux', null, true); | ||||
| 			process.exit(1); | ||||
| 		} | ||||
|  | ||||
| 		if (!await isPortAvailable(config.port)) { | ||||
| 			bootLogger.error(`Port ${config.port} is already in use`, null, true); | ||||
| 			process.exit(1); | ||||
| 		} | ||||
| 	} catch (e) { | ||||
| 		bootLogger.error('Fatal error occurred during initialization', null, true); | ||||
| 		process.exit(1); | ||||
| 	} | ||||
|  | ||||
| 	bootLogger.succ('Misskey initialized'); | ||||
|  | ||||
| 	if (!program.disableClustering) { | ||||
| 		await spawnWorkers(config.clusterLimit); | ||||
| 	} | ||||
|  | ||||
| 	if (!program.noDaemons) { | ||||
| 		require('../daemons/server-stats').default(); | ||||
| 		require('../daemons/notes-stats').default(); | ||||
| 		require('../daemons/queue-stats').default(); | ||||
| 	} | ||||
|  | ||||
| 	bootLogger.succ(`Now listening on port ${config.port} on ${config.url}`, null, true); | ||||
| } | ||||
|  | ||||
| const runningNodejsVersion = process.version.slice(1).split('.').map(x => parseInt(x, 10)); | ||||
| const requiredNodejsVersion = [11, 7, 0]; | ||||
| const satisfyNodejsVersion = !lessThan(runningNodejsVersion, requiredNodejsVersion); | ||||
|  | ||||
| function isWellKnownPort(port: number): boolean { | ||||
| 	return port < 1024; | ||||
| } | ||||
|  | ||||
| async function isPortAvailable(port: number): Promise<boolean> { | ||||
| 	return await portscanner.checkPortStatus(port, '127.0.0.1') === 'closed'; | ||||
| } | ||||
|  | ||||
| function showEnvironment(): void { | ||||
| 	const env = process.env.NODE_ENV; | ||||
| 	const logger = bootLogger.createSubLogger('env'); | ||||
| 	logger.info(typeof env == 'undefined' ? 'NODE_ENV is not set' : `NODE_ENV: ${env}`); | ||||
|  | ||||
| 	if (env !== 'production') { | ||||
| 		logger.warn('The environment is not in production mode.'); | ||||
| 		logger.warn('DO NOT USE FOR PRODUCTION PURPOSE!', null, true); | ||||
| 	} | ||||
|  | ||||
| 	logger.info(`You ${isRoot() ? '' : 'do not '}have root privileges`); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Init app | ||||
|  */ | ||||
| async function init(): Promise<Config> { | ||||
| 	showEnvironment(); | ||||
|  | ||||
| 	const nodejsLogger = bootLogger.createSubLogger('nodejs'); | ||||
|  | ||||
| 	nodejsLogger.info(`Version ${runningNodejsVersion.join('.')}`); | ||||
|  | ||||
| 	if (!satisfyNodejsVersion) { | ||||
| 		nodejsLogger.error(`Node.js version is less than ${requiredNodejsVersion.join('.')}. Please upgrade it.`, null, true); | ||||
| 		process.exit(1); | ||||
| 	} | ||||
|  | ||||
| 	await showMachineInfo(bootLogger); | ||||
|  | ||||
| 	const configLogger = bootLogger.createSubLogger('config'); | ||||
| 	let config; | ||||
|  | ||||
| 	try { | ||||
| 		config = loadConfig(); | ||||
| 	} catch (exception) { | ||||
| 		if (typeof exception === 'string') { | ||||
| 			configLogger.error(exception); | ||||
| 			process.exit(1); | ||||
| 		} | ||||
| 		if (exception.code === 'ENOENT') { | ||||
| 			configLogger.error('Configuration file not found', null, true); | ||||
| 			process.exit(1); | ||||
| 		} | ||||
| 		throw exception; | ||||
| 	} | ||||
|  | ||||
| 	configLogger.succ('Loaded'); | ||||
|  | ||||
| 	// Try to connect to DB | ||||
| 	try { | ||||
| 		bootLogger.info('Connecting database...'); | ||||
| 		await initDb(); | ||||
| 	} catch (e) { | ||||
| 		bootLogger.error('Cannot connect to database', null, true); | ||||
| 		bootLogger.error(e); | ||||
| 		process.exit(1); | ||||
| 	} | ||||
|  | ||||
| 	return config; | ||||
| } | ||||
|  | ||||
| async function spawnWorkers(limit: number = Infinity) { | ||||
| 	const workers = Math.min(limit, os.cpus().length); | ||||
| 	bootLogger.info(`Starting ${workers} worker${workers === 1 ? '' : 's'}...`); | ||||
| 	await Promise.all([...Array(workers)].map(spawnWorker)); | ||||
| 	bootLogger.succ('All workers started'); | ||||
| } | ||||
|  | ||||
| function spawnWorker(): Promise<void> { | ||||
| 	return new Promise(res => { | ||||
| 		const worker = cluster.fork(); | ||||
| 		worker.on('message', message => { | ||||
| 			if (message !== 'ready') return; | ||||
| 			res(); | ||||
| 		}); | ||||
| 	}); | ||||
| } | ||||
							
								
								
									
										20
									
								
								src/boot/worker.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								src/boot/worker.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| import * as cluster from 'cluster'; | ||||
| import { initDb } from '../db/postgre'; | ||||
|  | ||||
| /** | ||||
|  * Init worker process | ||||
|  */ | ||||
| export async function workerMain() { | ||||
| 	await initDb(); | ||||
|  | ||||
| 	// start server | ||||
| 	await require('../server').default(); | ||||
|  | ||||
| 	// start job queue | ||||
| 	require('../queue').default(); | ||||
|  | ||||
| 	if (cluster.isWorker) { | ||||
| 		// Send a 'ready' message to parent process | ||||
| 		process.send!('ready'); | ||||
| 	} | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="faExclamationCircle"/> {{ $t('title') }}</div> | ||||
| 		<template #title><fa :icon="faExclamationCircle"/> {{ $t('title') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<sequential-entrance animation="entranceFromTop" delay="25"> | ||||
| 				<div v-for="report in userReports" :key="report.id" class="haexwsjc"> | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa icon="broadcast-tower"/> {{ $t('announcements') }}</div> | ||||
| 		<template #title><fa :icon="faBroadcastTower"/> {{ $t('announcements') }}</template> | ||||
| 		<section v-for="(announcement, i) in announcements" class="fit-top"> | ||||
| 			<ui-input v-model="announcement.title" @change="save"> | ||||
| 				<span>{{ $t('title') }}</span> | ||||
| @@ -9,13 +9,16 @@ | ||||
| 			<ui-textarea v-model="announcement.text"> | ||||
| 				<span>{{ $t('text') }}</span> | ||||
| 			</ui-textarea> | ||||
| 			<ui-input v-model="announcement.image"> | ||||
| 				<span>{{ $t('image-url') }}</span> | ||||
| 			</ui-input> | ||||
| 			<ui-horizon-group class="fit-bottom"> | ||||
| 				<ui-button @click="save()"><fa :icon="['far', 'save']"/> {{ $t('save') }}</ui-button> | ||||
| 				<ui-button @click="remove(i)"><fa :icon="['far', 'trash-alt']"/> {{ $t('remove') }}</ui-button> | ||||
| 			</ui-horizon-group> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-button @click="add"><fa icon="plus"/> {{ $t('add') }}</ui-button> | ||||
| 			<ui-button @click="add"><fa :icon="faPlus"/> {{ $t('add') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
| </div> | ||||
| @@ -24,18 +27,20 @@ | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import i18n from '../../i18n'; | ||||
| import { faBroadcastTower, faPlus } from '@fortawesome/free-solid-svg-icons'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	i18n: i18n('admin/views/announcements.vue'), | ||||
| 	data() { | ||||
| 		return { | ||||
| 			announcements: [], | ||||
| 			faBroadcastTower, faPlus | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	created() { | ||||
| 		this.$root.getMeta().then(meta => { | ||||
| 			this.announcements = meta.broadcasts; | ||||
| 			this.announcements = meta.announcements; | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| @@ -43,7 +48,8 @@ export default Vue.extend({ | ||||
| 		add() { | ||||
| 			this.announcements.unshift({ | ||||
| 				title: '', | ||||
| 				text: '' | ||||
| 				text: '', | ||||
| 				image: null | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| @@ -65,7 +71,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 		save(silent) { | ||||
| 			this.$root.api('admin/update-meta', { | ||||
| 				broadcasts: this.announcements | ||||
| 				announcements: this.announcements | ||||
| 			}).then(() => { | ||||
| 				if (!silent) { | ||||
| 					this.$root.dialog({ | ||||
|   | ||||
| @@ -181,7 +181,12 @@ export default Vue.extend({ | ||||
| 				}, | ||||
| 				grid: { | ||||
| 					clipMarkers: false, | ||||
| 					borderColor: 'rgba(0, 0, 0, 0.1)' | ||||
| 					borderColor: 'rgba(0, 0, 0, 0.1)', | ||||
| 					xaxis: { | ||||
| 						lines: { | ||||
| 							show: true, | ||||
| 						} | ||||
| 					}, | ||||
| 				}, | ||||
| 				stroke: { | ||||
| 					curve: 'straight', | ||||
| @@ -240,6 +245,7 @@ export default Vue.extend({ | ||||
| 		federationInstancesChart(total: boolean): any { | ||||
| 			return { | ||||
| 				series: [{ | ||||
| 					name: 'Instances', | ||||
| 					data: this.format(total | ||||
| 						? this.stats.federation.instance.total | ||||
| 						: sum(this.stats.federation.instance.inc, negate(this.stats.federation.instance.dec)) | ||||
							
								
								
									
										196
									
								
								src/client/app/admin/views/dashboard.queue-charts.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										196
									
								
								src/client/app/admin/views/dashboard.queue-charts.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,196 @@ | ||||
| <template> | ||||
| <div class="mzxlfysy"> | ||||
| 	<div> | ||||
| 		<header> | ||||
| 			<span><fa :icon="faInbox"/> In</span> | ||||
| 			<span v-if="latestStats">{{ latestStats.inbox.activeSincePrevTick | number }} / {{ latestStats.inbox.delayed | number }}</span> | ||||
| 		</header> | ||||
| 		<div ref="in"></div> | ||||
| 	</div> | ||||
| 	<div> | ||||
| 		<header> | ||||
| 			<span><fa :icon="faPaperPlane"/> Out</span> | ||||
| 			<span v-if="latestStats">{{ latestStats.deliver.activeSincePrevTick | number }} / {{ latestStats.deliver.delayed | number }}</span> | ||||
| 		</header> | ||||
| 		<div ref="out"></div> | ||||
| 	</div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { faInbox } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faPaperPlane } from '@fortawesome/free-regular-svg-icons'; | ||||
| import ApexCharts from 'apexcharts'; | ||||
|  | ||||
| const limit = 150; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	data() { | ||||
| 		return { | ||||
| 			stats: [], | ||||
| 			inChart: null, | ||||
| 			outChart: null, | ||||
| 			faInbox, faPaperPlane | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	computed: { | ||||
| 		latestStats(): any { | ||||
| 			return this.stats[this.stats.length - 1]; | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	watch: { | ||||
| 		stats(stats) { | ||||
| 			this.inChart.updateSeries([{ | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.inbox.activeSincePrevTick })) | ||||
| 			}, { | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.inbox.active })) | ||||
| 			}, { | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.inbox.waiting })) | ||||
| 			}, { | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.inbox.delayed })) | ||||
| 			}]); | ||||
| 			this.outChart.updateSeries([{ | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.deliver.activeSincePrevTick })) | ||||
| 			}, { | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.deliver.active })) | ||||
| 			}, { | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.deliver.waiting })) | ||||
| 			}, { | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.deliver.delayed })) | ||||
| 			}]); | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		const chartOpts = { | ||||
| 			chart: { | ||||
| 				type: 'area', | ||||
| 				height: 200, | ||||
| 				animations: { | ||||
| 					dynamicAnimation: { | ||||
| 						enabled: false | ||||
| 					} | ||||
| 				}, | ||||
| 				toolbar: { | ||||
| 					show: false | ||||
| 				}, | ||||
| 				zoom: { | ||||
| 					enabled: false | ||||
| 				} | ||||
| 			}, | ||||
| 			dataLabels: { | ||||
| 				enabled: false | ||||
| 			}, | ||||
| 			grid: { | ||||
| 				clipMarkers: false, | ||||
| 				borderColor: 'rgba(0, 0, 0, 0.1)' | ||||
| 			}, | ||||
| 			stroke: { | ||||
| 				curve: 'straight', | ||||
| 				width: 2 | ||||
| 			}, | ||||
| 			tooltip: { | ||||
| 				enabled: false | ||||
| 			}, | ||||
| 			legend: { | ||||
| 				show: false | ||||
| 			}, | ||||
| 			colors: ['#00E396', '#00BCD4', '#FFB300', '#e53935'], | ||||
| 			series: [{ data: [] }, { data: [] }, { data: [] }, { data: [] }] as any, | ||||
| 			xaxis: { | ||||
| 				type: 'numeric', | ||||
| 				labels: { | ||||
| 					show: false | ||||
| 				}, | ||||
| 				tooltip: { | ||||
| 					enabled: false | ||||
| 				} | ||||
| 			}, | ||||
| 			yaxis: { | ||||
| 				show: false, | ||||
| 				min: 0, | ||||
| 			} | ||||
| 		}; | ||||
|  | ||||
| 		this.inChart = new ApexCharts(this.$refs.in, chartOpts); | ||||
| 		this.outChart = new ApexCharts(this.$refs.out, chartOpts); | ||||
|  | ||||
| 		this.inChart.render(); | ||||
| 		this.outChart.render(); | ||||
|  | ||||
| 		const connection = this.$root.stream.useSharedConnection('queueStats'); | ||||
| 		connection.on('stats', this.onStats); | ||||
| 		connection.on('statsLog', this.onStatsLog); | ||||
| 		connection.send('requestLog', { | ||||
| 			id: Math.random().toString().substr(2, 8), | ||||
| 			length: limit | ||||
| 		}); | ||||
|  | ||||
| 		this.$once('hook:beforeDestroy', () => { | ||||
| 			connection.dispose(); | ||||
| 			this.inChart.destroy(); | ||||
| 			this.outChart.destroy(); | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		onStats(stats) { | ||||
| 			this.stats.push(stats); | ||||
| 			if (this.stats.length > limit) this.stats.shift(); | ||||
| 		}, | ||||
|  | ||||
| 		onStatsLog(statsLog) { | ||||
| 			for (const stats of statsLog.reverse()) { | ||||
| 				this.onStats(stats); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .mzxlfysy | ||||
| 	display flex | ||||
|  | ||||
| 	> div | ||||
| 		display block | ||||
| 		flex 1 | ||||
| 		padding 20px 12px 0 12px | ||||
| 		box-shadow 0 2px 4px rgba(0, 0, 0, 0.1) | ||||
| 		background var(--face) | ||||
| 		border-radius 8px | ||||
|  | ||||
| 		&:first-child | ||||
| 			margin-right 16px | ||||
|  | ||||
| 		> header | ||||
| 			display flex | ||||
| 			padding 0 8px | ||||
| 			margin-bottom -16px | ||||
| 			color var(--adminDashboardCardFg) | ||||
| 			font-size 14px | ||||
|  | ||||
| 			> span | ||||
| 				&:last-child | ||||
| 					margin-left auto | ||||
| 					opacity 0.7 | ||||
|  | ||||
| 				> span | ||||
| 					opacity 0.7 | ||||
|  | ||||
| 		> div | ||||
| 			margin-bottom -10px | ||||
|  | ||||
| 	@media (max-width 1000px) | ||||
| 		display block | ||||
| 		margin-bottom 26px | ||||
|  | ||||
| 		> div | ||||
| 			&:first-child | ||||
| 				margin-right 0 | ||||
| 				margin-bottom 26px | ||||
|  | ||||
| </style> | ||||
| @@ -73,6 +73,10 @@ | ||||
| 		<x-charts ref="charts"/> | ||||
| 	</div> | ||||
|  | ||||
| 	<div class="queue"> | ||||
| 		<x-queue/> | ||||
| 	</div> | ||||
|  | ||||
| 	<div class="cpu-memory"> | ||||
| 		<x-cpu-memory :connection="connection"/> | ||||
| 	</div> | ||||
| @@ -86,9 +90,10 @@ | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import i18n from '../../i18n'; | ||||
| import XCpuMemory from "./cpu-memory.vue"; | ||||
| import XCharts from "./charts.vue"; | ||||
| import XApLog from "./ap-log.vue"; | ||||
| import XCpuMemory from "./dashboard.cpu-memory.vue"; | ||||
| import XQueue from "./dashboard.queue-charts.vue"; | ||||
| import XCharts from "./dashboard.charts.vue"; | ||||
| import XApLog from "./dashboard.ap-log.vue"; | ||||
| import { faDatabase } from '@fortawesome/free-solid-svg-icons'; | ||||
| import MarqueeText from 'vue-marquee-text-component'; | ||||
| import randomColor from 'randomcolor'; | ||||
| @@ -98,6 +103,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 	components: { | ||||
| 		XCpuMemory, | ||||
| 		XQueue, | ||||
| 		XCharts, | ||||
| 		XApLog, | ||||
| 		MarqueeText | ||||
| @@ -274,6 +280,9 @@ export default Vue.extend({ | ||||
| 	> .charts | ||||
| 		margin-bottom 16px | ||||
|  | ||||
| 	> .queue | ||||
| 		margin-bottom 16px | ||||
|  | ||||
| 	> .cpu-memory | ||||
| 		margin-bottom 16px | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="faTerminal"/> {{ $t('operation') }}</div> | ||||
| 		<template #title><fa :icon="faTerminal"/> {{ $t('operation') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-input v-model="target" type="text"> | ||||
| 				<span>{{ $t('fileid-or-url') }}</span> | ||||
| @@ -17,18 +17,18 @@ | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="faCloud"/> {{ $t('@.drive') }}</div> | ||||
| 		<template #title><fa :icon="faCloud"/> {{ $t('@.drive') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-horizon-group inputs> | ||||
| 				<ui-select v-model="sort"> | ||||
| 					<span slot="label">{{ $t('sort.title') }}</span> | ||||
| 					<template #label>{{ $t('sort.title') }}</template> | ||||
| 					<option value="-createdAt">{{ $t('sort.createdAtAsc') }}</option> | ||||
| 					<option value="+createdAt">{{ $t('sort.createdAtDesc') }}</option> | ||||
| 					<option value="-size">{{ $t('sort.sizeAsc') }}</option> | ||||
| 					<option value="+size">{{ $t('sort.sizeDesc') }}</option> | ||||
| 				</ui-select> | ||||
| 				<ui-select v-model="origin"> | ||||
| 					<span slot="label">{{ $t('origin.title') }}</span> | ||||
| 					<template #label>{{ $t('origin.title') }}</template> | ||||
| 					<option value="combined">{{ $t('origin.combined') }}</option> | ||||
| 					<option value="local">{{ $t('origin.local') }}</option> | ||||
| 					<option value="remote">{{ $t('origin.remote') }}</option> | ||||
| @@ -38,7 +38,7 @@ | ||||
| 				<div class="kidvdlkg" v-for="file in files"> | ||||
| 					<div @click="file._open = !file._open"> | ||||
| 						<div> | ||||
| 							<div class="thumbnail" :style="thumbnail(file)"></div> | ||||
| 							<x-file-thumbnail class="thumbnail" :file="file" fit="contain" @click="showFileMenu(file)"/> | ||||
| 						</div> | ||||
| 						<div> | ||||
| 							<header> | ||||
| @@ -48,7 +48,7 @@ | ||||
| 							<div> | ||||
| 								<div> | ||||
| 									<span style="margin-right:16px;">{{ file.type }}</span> | ||||
| 									<span>{{ file.datasize | bytes }}</span> | ||||
| 									<span>{{ file.size | bytes }}</span> | ||||
| 								</div> | ||||
| 								<div><mk-time :time="file.createdAt" mode="detail"/></div> | ||||
| 							</div> | ||||
| @@ -75,10 +75,15 @@ import Vue from 'vue'; | ||||
| import i18n from '../../i18n'; | ||||
| import { faCloud, faTerminal, faSearch } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faTrashAlt, faEye, faEyeSlash } from '@fortawesome/free-regular-svg-icons'; | ||||
| import XFileThumbnail from '../../common/views/components/drive-file-thumbnail.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	i18n: i18n('admin/views/drive.vue'), | ||||
|  | ||||
| 	components: { | ||||
| 		XFileThumbnail | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			file: null, | ||||
| @@ -151,13 +156,6 @@ export default Vue.extend({ | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		thumbnail(file: any): any { | ||||
| 			return { | ||||
| 				'background-color': file.properties.avgColor && file.properties.avgColor.length == 3 ? `rgb(${file.properties.avgColor.join(',')})` : 'transparent', | ||||
| 				'background-image': `url(${file.thumbnailUrl})` | ||||
| 			}; | ||||
| 		}, | ||||
|  | ||||
| 		async del(file: any) { | ||||
| 			const process = async () => { | ||||
| 				await this.$root.api('drive/files/delete', { fileId: file.id }); | ||||
| @@ -179,9 +177,9 @@ export default Vue.extend({ | ||||
| 			this.$root.api('drive/files/update', { | ||||
| 				fileId: file.id, | ||||
| 				isSensitive: !file.isSensitive | ||||
| 			}).then(() => { | ||||
| 				file.isSensitive = !file.isSensitive; | ||||
| 			}); | ||||
|  | ||||
| 			file.isSensitive = !file.isSensitive; | ||||
| 		}, | ||||
|  | ||||
| 		async show() { | ||||
| @@ -244,7 +242,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 		> div:nth-child(1) | ||||
| 			> .thumbnail | ||||
| 				display block | ||||
| 				display flex | ||||
| 				width 64px | ||||
| 				height 64px | ||||
| 				background-size cover | ||||
|   | ||||
| @@ -1,20 +1,20 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa icon="plus"/> {{ $t('add-emoji.title') }}</div> | ||||
| 		<template #title><fa icon="plus"/> {{ $t('add-emoji.title') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-horizon-group inputs> | ||||
| 				<ui-input v-model="name"> | ||||
| 					<span>{{ $t('add-emoji.name') }}</span> | ||||
| 					<span slot="desc">{{ $t('add-emoji.name-desc') }}</span> | ||||
| 					<template #desc>{{ $t('add-emoji.name-desc') }}</template> | ||||
| 				</ui-input> | ||||
| 				<ui-input v-model="aliases"> | ||||
| 					<span>{{ $t('add-emoji.aliases') }}</span> | ||||
| 					<span slot="desc">{{ $t('add-emoji.aliases-desc') }}</span> | ||||
| 					<template #desc>{{ $t('add-emoji.aliases-desc') }}</template> | ||||
| 				</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 			<ui-input v-model="url"> | ||||
| 				<i slot="icon"><fa icon="link"/></i> | ||||
| 				<template #icon><fa icon="link"/></template> | ||||
| 				<span>{{ $t('add-emoji.url') }}</span> | ||||
| 			</ui-input> | ||||
| 			<ui-info>{{ $t('add-emoji.info') }}</ui-info> | ||||
| @@ -23,7 +23,7 @@ | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="faGrin"/> {{ $t('emojis.title') }}</div> | ||||
| 		<template #title><fa :icon="faGrin"/> {{ $t('emojis.title') }}</template> | ||||
| 		<section v-for="emoji in emojis" class="oryfrbft"> | ||||
| 			<div> | ||||
| 				<img :src="emoji.url" :alt="emoji.name" style="width: 64px;"/> | ||||
| @@ -38,7 +38,7 @@ | ||||
| 					</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-input v-model="emoji.url"> | ||||
| 					<i slot="icon"><fa icon="link"/></i> | ||||
| 					<template #icon><fa icon="link"/></template> | ||||
| 					<span>{{ $t('add-emoji.url') }}</span> | ||||
| 				</ui-input> | ||||
| 				<ui-horizon-group class="fit-bottom"> | ||||
|   | ||||
| @@ -1,45 +1,59 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="faTerminal"/> {{ $t('federation') }}</div> | ||||
| 		<template #title><fa :icon="faTerminal"/> {{ $t('instance') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-input class="target" v-model="target" type="text" @enter="showInstance()"> | ||||
| 				<span>{{ $t('host') }}</span> | ||||
| 				<template #prefix><fa :icon="faServer"/></template> | ||||
| 			</ui-input> | ||||
| 			<ui-button @click="showInstance()"><fa :icon="faSearch"/> {{ $t('lookup') }}</ui-button> | ||||
|  | ||||
| 			<div class="instance" v-if="instance"> | ||||
| 				<ui-input :value="instance.host" type="text" readonly> | ||||
| 					<span>{{ $t('host') }}</span> | ||||
| 				</ui-input> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input :value="instance.host" type="text" readonly> | ||||
| 						<span>{{ $t('host') }}</span> | ||||
| 						<template #prefix><fa :icon="faServer"/></template> | ||||
| 					</ui-input> | ||||
| 					<ui-input :value="instance.caughtAt | date" type="text" readonly> | ||||
| 						<span>{{ $t('caught-at') }}</span> | ||||
| 						<template #prefix><fa :icon="faCrosshairs"/></template> | ||||
| 					</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input :value="instance.notesCount | number" type="text" readonly> | ||||
| 						<span>{{ $t('notes') }}</span> | ||||
| 						<template #prefix><fa :icon="faEnvelopeOpenText"/></template> | ||||
| 					</ui-input> | ||||
| 					<ui-input :value="instance.usersCount | number" type="text" readonly> | ||||
| 						<span>{{ $t('users') }}</span> | ||||
| 						<template #prefix><fa :icon="faUsers"/></template> | ||||
| 					</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input :value="instance.followingCount | number" type="text" readonly> | ||||
| 						<span>{{ $t('following') }}</span> | ||||
| 						<template #prefix><fa :icon="faCaretDown"/></template> | ||||
| 					</ui-input> | ||||
| 					<ui-input :value="instance.followersCount | number" type="text" readonly> | ||||
| 						<span>{{ $t('followers') }}</span> | ||||
| 						<template #prefix><fa :icon="faCaretUp"/></template> | ||||
| 					</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input :value="instance.latestRequestSentAt" type="text" readonly> | ||||
| 					<ui-input :value="instance.latestRequestSentAt | date" type="text" readonly> | ||||
| 						<span>{{ $t('latest-request-sent-at') }}</span> | ||||
| 						<template #prefix><fa :icon="faPaperPlane"/></template> | ||||
| 					</ui-input> | ||||
| 					<ui-input :value="instance.latestStatus" type="text" readonly> | ||||
| 						<span>{{ $t('status') }}</span> | ||||
| 						<template #prefix><fa :icon="faTrafficLight"/></template> | ||||
| 					</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-input :value="instance.latestRequestReceivedAt" type="text" readonly> | ||||
| 				<ui-input :value="instance.latestRequestReceivedAt | date" type="text" readonly> | ||||
| 					<span>{{ $t('latest-request-received-at') }}</span> | ||||
| 					<template #prefix><fa :icon="faInbox"/></template> | ||||
| 				</ui-input> | ||||
| 				<ui-switch v-model="instance.isBlocked" @change="updateInstance()">{{ $t('block') }}</ui-switch> | ||||
| 				<ui-switch v-model="instance.isMarkedAsClosed" @change="updateInstance()">{{ $t('marked-as-closed') }}</ui-switch> | ||||
| 				<details> | ||||
| 					<summary>{{ $t('charts') }}</summary> | ||||
| @@ -64,6 +78,10 @@ | ||||
| 					</ui-horizon-group> | ||||
| 					<div ref="chart"></div> | ||||
| 				</details> | ||||
| 				<details> | ||||
| 					<summary>{{ $t('delete-all-files') }}</summary> | ||||
| 					<ui-button @click="deleteAllFiles()" style="margin-top: 16px;"><fa :icon="faTrashAlt"/> {{ $t('delete-all-files') }}</ui-button> | ||||
| 				</details> | ||||
| 				<details> | ||||
| 					<summary>{{ $t('remove-all-following') }}</summary> | ||||
| 					<ui-button @click="removeAllFollowing()" style="margin-top: 16px;"><fa :icon="faMinusCircle"/> {{ $t('remove-all-following') }}</ui-button> | ||||
| @@ -74,11 +92,11 @@ | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="faServer"/> {{ $t('instances') }}</div> | ||||
| 		<template #title><fa :icon="faServer"/> {{ $t('instances') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-horizon-group inputs> | ||||
| 				<ui-select v-model="sort"> | ||||
| 					<span slot="label">{{ $t('sort') }}</span> | ||||
| 					<template #label>{{ $t('sort') }}</template> | ||||
| 					<option value="-caughtAt">{{ $t('sorts.caughtAtAsc') }}</option> | ||||
| 					<option value="+caughtAt">{{ $t('sorts.caughtAtDesc') }}</option> | ||||
| 					<option value="-lastCommunicatedAt">{{ $t('sorts.lastCommunicatedAtAsc') }}</option> | ||||
| @@ -97,7 +115,7 @@ | ||||
| 					<option value="+driveFiles">{{ $t('sorts.driveFilesDesc') }}</option> | ||||
| 				</ui-select> | ||||
| 				<ui-select v-model="state"> | ||||
| 					<span slot="label">{{ $t('state') }}</span> | ||||
| 					<template #label>{{ $t('state') }}</template> | ||||
| 					<option value="all">{{ $t('states.all') }}</option> | ||||
| 					<option value="blocked">{{ $t('states.blocked') }}</option> | ||||
| 					<option value="notResponding">{{ $t('states.not-responding') }}</option> | ||||
| @@ -115,7 +133,7 @@ | ||||
| 					<span>{{ $t('status') }}</span> | ||||
| 				</header> | ||||
| 				<div v-for="instance in instances" :style="{ opacity: instance.isNotResponding ? 0.5 : 1 }"> | ||||
| 					<a @click.prevent="showInstance(instance.host)" target="_blank" :href="`https://${instance.host}`" :style="{ textDecoration: instance.isMarkedAsClosed ? 'line-through' : 'none' }">{{ instance.host }}</a> | ||||
| 					<a @click.prevent="showInstance(instance.host)" rel="nofollow noopener" target="_blank" :href="`https://${instance.host}`" :style="{ textDecoration: instance.isMarkedAsClosed ? 'line-through' : 'none' }">{{ instance.host }}</a> | ||||
| 					<span>{{ instance.notesCount | number }}</span> | ||||
| 					<span>{{ instance.usersCount | number }}</span> | ||||
| 					<span>{{ instance.followingCount | number }}</span> | ||||
| @@ -127,13 +145,24 @@ | ||||
| 			<ui-info v-if="instances.length == limit">{{ $t('result-is-truncated', { n: limit }) }}</ui-info> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<template #title><fa :icon="faBan"/> {{ $t('blocked-hosts') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-textarea v-model="blockedHosts"> | ||||
| 				<template #desc>{{ $t('blocked-hosts-info') }}</template> | ||||
| 			</ui-textarea> | ||||
| 			<ui-button @click="saveBlockedHosts">{{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import i18n from '../../i18n'; | ||||
| import { faGlobe, faTerminal, faSearch, faMinusCircle, faServer } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faPaperPlane } from '@fortawesome/free-regular-svg-icons'; | ||||
| import { faTrashAlt, faBan, faGlobe, faTerminal, faSearch, faMinusCircle, faServer, faCrosshairs, faEnvelopeOpenText, faUsers, faCaretDown, faCaretUp, faTrafficLight, faInbox } from '@fortawesome/free-solid-svg-icons'; | ||||
| import ApexCharts from 'apexcharts'; | ||||
| import * as tinycolor from 'tinycolor2'; | ||||
|  | ||||
| @@ -144,19 +173,24 @@ const negate = arr => arr.map(x => -x); | ||||
| export default Vue.extend({ | ||||
| 	i18n: i18n('admin/views/federation.vue'), | ||||
|  | ||||
| 	filters: { | ||||
| 		date: v => v ? new Date(v).toLocaleString() : 'N/A' | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			instance: null, | ||||
| 			target: null, | ||||
| 			sort: '+lastCommunicatedAt', | ||||
| 			state: 'all', | ||||
| 			limit: 50, | ||||
| 			limit: 100, | ||||
| 			instances: [], | ||||
| 			chart: null, | ||||
| 			chartSrc: 'requests', | ||||
| 			chartSpan: 'hour', | ||||
| 			chartInstance: null, | ||||
| 			faGlobe, faTerminal, faSearch, faMinusCircle, faServer | ||||
| 			blockedHosts: '', | ||||
| 			faTrashAlt, faBan, faGlobe, faTerminal, faSearch, faMinusCircle, faServer, faCrosshairs, faEnvelopeOpenText, faUsers, faCaretDown, faCaretUp, faPaperPlane, faTrafficLight, faInbox | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| @@ -226,6 +260,10 @@ export default Vue.extend({ | ||||
|  | ||||
| 	mounted() { | ||||
| 		this.fetchInstances(); | ||||
|  | ||||
| 		this.$root.getMeta().then(meta => { | ||||
| 			this.blockedHosts = meta.blockedHosts.join('\n'); | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| 	beforeDestroy() { | ||||
| @@ -273,6 +311,17 @@ export default Vue.extend({ | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		deleteAllFiles() { | ||||
| 			this.$root.api('admin/federation/delete-all-files', { | ||||
| 				host: this.instance.host | ||||
| 			}).then(() => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					splash: true | ||||
| 				}); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		updateInstance() { | ||||
| 			this.$root.api('admin/federation/update-instance', { | ||||
| 				host: this.instance.host, | ||||
| @@ -457,6 +506,22 @@ export default Vue.extend({ | ||||
| 				}] | ||||
| 			}; | ||||
| 		}, | ||||
|  | ||||
| 		saveBlockedHosts() { | ||||
| 			this.$root.api('admin/update-meta', { | ||||
| 				blockedHosts: this.blockedHosts.split('\n') | ||||
| 			}).then(() => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					text: this.$t('saved') | ||||
| 				}); | ||||
| 			}).catch(e => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'error', | ||||
| 					text: e | ||||
| 				}); | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|   | ||||
| @@ -1,41 +0,0 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title">{{ $t('hided-tags') }}</div> | ||||
| 		<section> | ||||
| 			<textarea class="jdnqwkzlnxcfftthoybjxrebyolvoucw" v-model="hidedTags"></textarea> | ||||
| 			<ui-button @click="save">{{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import i18n from '../../i18n'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	i18n: i18n('admin/views/hashtags.vue'), | ||||
| 	data() { | ||||
| 		return { | ||||
| 			hidedTags: '', | ||||
| 		}; | ||||
| 	}, | ||||
| 	created() { | ||||
| 		this.$root.getMeta().then(meta => { | ||||
| 			this.hidedTags = meta.hidedTags.join('\n'); | ||||
| 		}); | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		save() { | ||||
| 			this.$root.api('admin/update-meta', { | ||||
| 				hidedTags: this.hidedTags.split('\n') | ||||
| 			}).then(() => { | ||||
| 				//this.$root.os.apis.dialog({ text: `Saved` }); | ||||
| 			}).catch(e => { | ||||
| 				//this.$root.os.apis.dialog({ text: `Failed ${e}` }); | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
| @@ -21,13 +21,13 @@ | ||||
| 			<li @click="nav('dashboard')" :class="{ active: page == 'dashboard' }"><fa icon="home" fixed-width/>{{ $t('dashboard') }}</li> | ||||
| 			<li @click="nav('instance')" :class="{ active: page == 'instance' }"><fa icon="cog" fixed-width/>{{ $t('instance') }}</li> | ||||
| 			<li @click="nav('queue')" :class="{ active: page == 'queue' }"><fa :icon="faTasks" fixed-width/>{{ $t('queue') }}</li> | ||||
| 			<li @click="nav('logs')" :class="{ active: page == 'logs' }"><fa :icon="faStream" fixed-width/>{{ $t('logs') }}</li> | ||||
| 			<li @click="nav('moderators')" :class="{ active: page == 'moderators' }"><fa :icon="faHeadset" fixed-width/>{{ $t('moderators') }}</li> | ||||
| 			<li @click="nav('users')" :class="{ active: page == 'users' }"><fa icon="users" fixed-width/>{{ $t('users') }}</li> | ||||
| 			<li @click="nav('drive')" :class="{ active: page == 'drive' }"><fa icon="cloud" fixed-width/>{{ $t('@.drive') }}</li> | ||||
| 			<li @click="nav('federation')" :class="{ active: page == 'federation' }"><fa :icon="faGlobe" fixed-width/>{{ $t('federation') }}</li> | ||||
| 			<li @click="nav('emoji')" :class="{ active: page == 'emoji' }"><fa :icon="faGrin" fixed-width/>{{ $t('emoji') }}</li> | ||||
| 			<li @click="nav('announcements')" :class="{ active: page == 'announcements' }"><fa icon="broadcast-tower" fixed-width/>{{ $t('announcements') }}</li> | ||||
| 			<li @click="nav('hashtags')" :class="{ active: page == 'hashtags' }"><fa icon="hashtag" fixed-width/>{{ $t('hashtags') }}</li> | ||||
| 			<li @click="nav('abuse')" :class="{ active: page == 'abuse' }"><fa :icon="faExclamationCircle" fixed-width/>{{ $t('abuse') }}</li> | ||||
| 		</ul> | ||||
| 		<div class="back-to-misskey"> | ||||
| @@ -42,11 +42,11 @@ | ||||
| 			<div v-if="page == 'dashboard'"><x-dashboard/></div> | ||||
| 			<div v-if="page == 'instance'"><x-instance/></div> | ||||
| 			<div v-if="page == 'queue'"><x-queue/></div> | ||||
| 			<div v-if="page == 'logs'"><x-logs/></div> | ||||
| 			<div v-if="page == 'moderators'"><x-moderators/></div> | ||||
| 			<div v-if="page == 'users'"><x-users/></div> | ||||
| 			<div v-if="page == 'emoji'"><x-emoji/></div> | ||||
| 			<div v-if="page == 'announcements'"><x-announcements/></div> | ||||
| 			<div v-if="page == 'hashtags'"><x-hashtags/></div> | ||||
| 			<div v-if="page == 'drive'"><x-drive/></div> | ||||
| 			<div v-if="page == 'federation'"><x-federation/></div> | ||||
| 			<div v-if="page == 'abuse'"><x-abuse/></div> | ||||
| @@ -62,16 +62,16 @@ import { version } from '../../config'; | ||||
| import XDashboard from "./dashboard.vue"; | ||||
| import XInstance from "./instance.vue"; | ||||
| import XQueue from "./queue.vue"; | ||||
| import XLogs from "./logs.vue"; | ||||
| import XModerators from "./moderators.vue"; | ||||
| import XEmoji from "./emoji.vue"; | ||||
| import XAnnouncements from "./announcements.vue"; | ||||
| import XHashtags from "./hashtags.vue"; | ||||
| import XUsers from "./users.vue"; | ||||
| import XDrive from "./drive.vue"; | ||||
| import XAbuse from "./abuse.vue"; | ||||
| import XFederation from "./federation.vue"; | ||||
|  | ||||
| import { faHeadset, faArrowLeft, faGlobe, faExclamationCircle, faTasks } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faHeadset, faArrowLeft, faGlobe, faExclamationCircle, faTasks, faStream } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faGrin } from '@fortawesome/free-regular-svg-icons'; | ||||
|  | ||||
| // Detect the user agent | ||||
| @@ -84,10 +84,10 @@ export default Vue.extend({ | ||||
| 		XDashboard, | ||||
| 		XInstance, | ||||
| 		XQueue, | ||||
| 		XLogs, | ||||
| 		XModerators, | ||||
| 		XEmoji, | ||||
| 		XAnnouncements, | ||||
| 		XHashtags, | ||||
| 		XUsers, | ||||
| 		XDrive, | ||||
| 		XAbuse, | ||||
| @@ -107,7 +107,8 @@ export default Vue.extend({ | ||||
| 			faHeadset, | ||||
| 			faGlobe, | ||||
| 			faExclamationCircle, | ||||
| 			faTasks | ||||
| 			faTasks, | ||||
| 			faStream | ||||
| 		}; | ||||
| 	}, | ||||
| 	methods: { | ||||
|   | ||||
| @@ -1,135 +1,242 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa icon="cog"/> {{ $t('instance') }}</div> | ||||
| 		<section class="fit-top fit-bottom"> | ||||
| 		<template #title><fa icon="cog"/> {{ $t('instance') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-input :value="host" readonly>{{ $t('host') }}</ui-input> | ||||
| 			<ui-input v-model="name">{{ $t('instance-name') }}</ui-input> | ||||
| 			<ui-textarea v-model="description">{{ $t('instance-description') }}</ui-textarea> | ||||
| 			<ui-input v-model="mascotImageUrl"><i slot="icon"><fa icon="link"/></i>{{ $t('logo-url') }}</ui-input> | ||||
| 			<ui-input v-model="bannerUrl"><i slot="icon"><fa icon="link"/></i>{{ $t('banner-url') }}</ui-input> | ||||
| 			<ui-input v-model="errorImageUrl"><i slot="icon"><fa icon="link"/></i>{{ $t('error-image-url') }}</ui-input> | ||||
| 			<ui-input v-model="languages"><i slot="icon"><fa icon="language"/></i>{{ $t('languages') }}<span slot="desc">{{ $t('languages-desc') }}</span></ui-input> | ||||
| 			<ui-input v-model="iconUrl"><template #icon><fa icon="link"/></template>{{ $t('icon-url') }}</ui-input> | ||||
| 			<ui-input v-model="mascotImageUrl"><template #icon><fa icon="link"/></template>{{ $t('logo-url') }}</ui-input> | ||||
| 			<ui-input v-model="bannerUrl"><template #icon><fa icon="link"/></template>{{ $t('banner-url') }}</ui-input> | ||||
| 			<ui-input v-model="errorImageUrl"><template #icon><fa icon="link"/></template>{{ $t('error-image-url') }}</ui-input> | ||||
| 			<ui-input v-model="ToSUrl"><template #icon><fa icon="link"/></template>{{ $t('tos-url') }}</ui-input> | ||||
| 			<ui-input v-model="languages"><template #icon><fa icon="language"/></template>{{ $t('languages') }}<template #desc>{{ $t('languages-desc') }}</template></ui-input> | ||||
| 			<details> | ||||
| 				<summary>{{ $t('advanced-config') }}</summary> | ||||
| 				<ui-input v-model="repositoryUrl"><template #icon><fa icon="link"/></template>{{ $t('repository-url') }}</ui-input> | ||||
| 				<ui-input v-model="feedbackUrl"><template #icon><fa icon="link"/></template>{{ $t('feedback-url') }}</ui-input> | ||||
| 			</details> | ||||
| 		</section> | ||||
| 		<section class="fit-bottom"> | ||||
| 			<header><fa :icon="faHeadset"/> {{ $t('maintainer-config') }}</header> | ||||
| 			<ui-input v-model="maintainerName">{{ $t('maintainer-name') }}</ui-input> | ||||
| 			<ui-input v-model="maintainerEmail" type="email"><i slot="icon"><fa :icon="farEnvelope"/></i>{{ $t('maintainer-email') }}</ui-input> | ||||
| 			<ui-input v-model="maintainerEmail" type="email"><template #icon><fa :icon="farEnvelope"/></template>{{ $t('maintainer-email') }}</ui-input> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-switch v-model="disableRegistration">{{ $t('disable-registration') }}</ui-switch> | ||||
| 			<ui-button v-if="disableRegistration" @click="invite">{{ $t('invite') }}</ui-button> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<template #title><fa :icon="faPencilAlt"/> {{ $t('note-and-tl') }}</template> | ||||
| 		<section class="fit-top fit-bottom"> | ||||
| 			<ui-input v-model="maxNoteTextLength">{{ $t('max-note-text-length') }}</ui-input> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-switch v-model="disableRegistration">{{ $t('disable-registration') }}</ui-switch> | ||||
| 			<ui-switch v-model="disableLocalTimeline">{{ $t('disable-local-timeline') }}</ui-switch> | ||||
| 			<ui-switch v-model="disableGlobalTimeline">{{ $t('disable-global-timeline') }}</ui-switch> | ||||
| 			<ui-info>{{ $t('disabling-timelines-info') }}</ui-info> | ||||
| 		</section> | ||||
| 		<section class="fit-bottom"> | ||||
| 			<header><fa icon="cloud"/> {{ $t('drive-config') }}</header> | ||||
| 			<ui-switch v-model="cacheRemoteFiles">{{ $t('cache-remote-files') }}<span slot="desc">{{ $t('cache-remote-files-desc') }}</span></ui-switch> | ||||
| 			<ui-input v-model="localDriveCapacityMb" type="number">{{ $t('local-drive-capacity-mb') }}<span slot="suffix">MB</span><span slot="desc">{{ $t('mb') }}</span></ui-input> | ||||
| 			<ui-input v-model="remoteDriveCapacityMb" type="number" :disabled="!cacheRemoteFiles">{{ $t('remote-drive-capacity-mb') }}<span slot="suffix">MB</span><span slot="desc">{{ $t('mb') }}</span></ui-input> | ||||
| 		</section> | ||||
| 		<section class="fit-bottom"> | ||||
| 			<header><fa :icon="faShieldAlt"/> {{ $t('recaptcha-config') }}</header> | ||||
| 			<ui-switch v-model="enableRecaptcha">{{ $t('enable-recaptcha') }}</ui-switch> | ||||
| 			<ui-info>{{ $t('recaptcha-info') }}</ui-info> | ||||
| 			<ui-horizon-group inputs> | ||||
| 				<ui-input v-model="recaptchaSiteKey" :disabled="!enableRecaptcha"><i slot="icon"><fa icon="key"/></i>{{ $t('recaptcha-site-key') }}</ui-input> | ||||
| 				<ui-input v-model="recaptchaSecretKey" :disabled="!enableRecaptcha"><i slot="icon"><fa icon="key"/></i>{{ $t('recaptcha-secret-key') }}</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 		<section> | ||||
| 			<ui-switch v-model="enableEmojiReaction">{{ $t('enable-emoji-reaction') }}</ui-switch> | ||||
| 			<ui-switch v-model="useStarForReactionFallback">{{ $t('use-star-for-reaction-fallback') }}</ui-switch> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<header><fa :icon="faGhost"/> {{ $t('proxy-account-config') }}</header> | ||||
| 			<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<template #title><fa icon="cloud"/> {{ $t('drive-config') }}</template> | ||||
| 		<section> | ||||
| 			<ui-switch v-model="useObjectStorage">{{ $t('use-object-storage') }}</ui-switch> | ||||
| 			<template v-if="useObjectStorage"> | ||||
| 				<ui-info> | ||||
| 					<i18n path="object-storage-s3-info"> | ||||
| 						<a href="https://docs.aws.amazon.com/general/latest/gr/rande.html" target="_blank">{{ $t('object-storage-s3-info-here') }}</a> | ||||
| 					</i18n> | ||||
| 				</ui-info> | ||||
| 				<ui-info>{{ $t('object-storage-gcs-info') }}</ui-info> | ||||
| 				<ui-input v-model="objectStorageBaseUrl" :disabled="!useObjectStorage">{{ $t('object-storage-base-url') }}</ui-input> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input v-model="objectStorageBucket" :disabled="!useObjectStorage">{{ $t('object-storage-bucket') }}</ui-input> | ||||
| 					<ui-input v-model="objectStoragePrefix" :disabled="!useObjectStorage">{{ $t('object-storage-prefix') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-input v-model="objectStorageEndpoint" :disabled="!useObjectStorage">{{ $t('object-storage-endpoint') }}</ui-input> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input v-model="objectStorageRegion" :disabled="!useObjectStorage">{{ $t('object-storage-region') }}</ui-input> | ||||
| 					<ui-input v-model="objectStoragePort" type="number" :disabled="!useObjectStorage">{{ $t('object-storage-port') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input v-model="objectStorageAccessKey" :disabled="!useObjectStorage"><template #icon><fa icon="key"/></template>{{ $t('object-storage-access-key') }}</ui-input> | ||||
| 					<ui-input v-model="objectStorageSecretKey" :disabled="!useObjectStorage"><template #icon><fa icon="key"/></template>{{ $t('object-storage-secret-key') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-switch v-model="objectStorageUseSSL" :disabled="!useObjectStorage">{{ $t('object-storage-use-ssl') }}</ui-switch> | ||||
| 			</template> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-switch v-model="cacheRemoteFiles">{{ $t('cache-remote-files') }}<template #desc>{{ $t('cache-remote-files-desc') }}</template></ui-switch> | ||||
| 		</section> | ||||
| 		<section class="fit-top fit-bottom"> | ||||
| 			<ui-input v-model="localDriveCapacityMb" type="number">{{ $t('local-drive-capacity-mb') }}<template #suffix>MB</template><template #desc>{{ $t('mb') }}</template></ui-input> | ||||
| 			<ui-input v-model="remoteDriveCapacityMb" type="number" :disabled="!cacheRemoteFiles">{{ $t('remote-drive-capacity-mb') }}<template #suffix>MB</template><template #desc>{{ $t('mb') }}</template></ui-input> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<template #title><fa :icon="faThumbtack"/> {{ $t('pinned-users') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-textarea v-model="pinnedUsers"> | ||||
| 				<template #desc>{{ $t('pinned-users-info') }}</template> | ||||
| 			</ui-textarea> | ||||
| 			<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<template #title><fa :icon="faGhost"/> {{ $t('proxy-account-config') }}</template> | ||||
| 		<section> | ||||
| 			<ui-info>{{ $t('proxy-account-info') }}</ui-info> | ||||
| 			<ui-input v-model="proxyAccount"><span slot="prefix">@</span>{{ $t('proxy-account-username') }}<span slot="desc">{{ $t('proxy-account-username-desc') }}</span></ui-input> | ||||
| 			<ui-input v-model="proxyAccount"><template #prefix>@</template>{{ $t('proxy-account-username') }}<template #desc>{{ $t('proxy-account-username-desc') }}</template></ui-input> | ||||
| 			<ui-info warn>{{ $t('proxy-account-warn') }}</ui-info> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<header><fa :icon="farEnvelope"/> {{ $t('email-config') }}</header> | ||||
| 			<ui-switch v-model="enableEmail">{{ $t('enable-email') }}<span slot="desc">{{ $t('email-config-info') }}</span></ui-switch> | ||||
| 			<ui-input v-model="email" type="email" :disabled="!enableEmail">{{ $t('email') }}</ui-input> | ||||
| 			<ui-horizon-group inputs> | ||||
| 				<ui-input v-model="smtpHost" :disabled="!enableEmail">{{ $t('smtp-host') }}</ui-input> | ||||
| 				<ui-input v-model="smtpPort" type="number" :disabled="!enableEmail">{{ $t('smtp-port') }}</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 			<ui-horizon-group inputs> | ||||
| 				<ui-input v-model="smtpUser" :disabled="!enableEmail">{{ $t('smtp-user') }}</ui-input> | ||||
| 				<ui-input v-model="smtpPass" type="password" :withPasswordToggle="true" :disabled="!enableEmail">{{ $t('smtp-pass') }}</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 			<ui-switch v-model="smtpSecure" :disabled="!enableEmail">{{ $t('smtp-secure') }}<span slot="desc">{{ $t('smtp-secure-info') }}</span></ui-switch> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<header><fa :icon="faBolt"/> {{ $t('serviceworker-config') }}</header> | ||||
| 			<ui-switch v-model="enableServiceWorker">{{ $t('enable-serviceworker') }}<span slot="desc">{{ $t('serviceworker-info') }}</span></ui-switch> | ||||
| 			<ui-info>{{ $t('vapid-info') }}<br><code>npm i web-push -g<br>web-push generate-vapid-keys</code></ui-info> | ||||
| 			<ui-horizon-group inputs class="fit-bottom"> | ||||
| 				<ui-input v-model="swPublicKey" :disabled="!enableServiceWorker"><i slot="icon"><fa icon="key"/></i>{{ $t('vapid-publickey') }}</ui-input> | ||||
| 				<ui-input v-model="swPrivateKey" :disabled="!enableServiceWorker"><i slot="icon"><fa icon="key"/></i>{{ $t('vapid-privatekey') }}</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<header>summaly Proxy</header> | ||||
| 			<ui-input v-model="summalyProxy">URL</ui-input> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<header><fa :icon="faUserPlus"/> {{ $t('user-recommendation-config') }}</header> | ||||
| 			<ui-switch v-model="enableExternalUserRecommendation">{{ $t('enable-external-user-recommendation') }}</ui-switch> | ||||
| 			<ui-input v-model="externalUserRecommendationEngine" :disabled="!enableExternalUserRecommendation">{{ $t('external-user-recommendation-engine') }}<span slot="desc">{{ $t('external-user-recommendation-engine-desc') }}</span></ui-input> | ||||
| 			<ui-input v-model="externalUserRecommendationTimeout" type="number" :disabled="!enableExternalUserRecommendation">{{ $t('external-user-recommendation-timeout') }}<span slot="suffix">ms</span><span slot="desc">{{ $t('external-user-recommendation-timeout-desc') }}</span></ui-input> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-button @click="updateMeta">{{ $t('save') }}</ui-button> | ||||
| 			<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<div slot="title">{{ $t('invite') }}</div> | ||||
| 		<template #title><fa :icon="farEnvelope"/> {{ $t('email-config') }}</template> | ||||
| 		<section> | ||||
| 			<ui-button @click="invite">{{ $t('invite') }}</ui-button> | ||||
| 			<p v-if="inviteCode">Code: <code>{{ inviteCode }}</code></p> | ||||
| 			<ui-switch v-model="enableEmail">{{ $t('enable-email') }}<template #desc>{{ $t('email-config-info') }}</template></ui-switch> | ||||
| 			<template v-if="enableEmail"> | ||||
| 				<ui-input v-model="email" type="email" :disabled="!enableEmail">{{ $t('email') }}</ui-input> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input v-model="smtpHost" :disabled="!enableEmail">{{ $t('smtp-host') }}</ui-input> | ||||
| 					<ui-input v-model="smtpPort" type="number" :disabled="!enableEmail">{{ $t('smtp-port') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-switch v-model="smtpAuth">{{ $t('smtp-auth') }}</ui-switch> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input v-model="smtpUser" :disabled="!enableEmail || !smtpAuth">{{ $t('smtp-user') }}</ui-input> | ||||
| 					<ui-input v-model="smtpPass" type="password" :with-password-toggle="true" :disabled="!enableEmail || !smtpAuth">{{ $t('smtp-pass') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-switch v-model="smtpSecure" :disabled="!enableEmail">{{ $t('smtp-secure') }}<template #desc>{{ $t('smtp-secure-info') }}</template></ui-switch> | ||||
| 			</template> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="['fab', 'twitter']"/> {{ $t('twitter-integration-config') }}</div> | ||||
| 		<template #title><fa :icon="faBolt"/> {{ $t('serviceworker-config') }}</template> | ||||
| 		<section> | ||||
| 			<ui-switch v-model="enableServiceWorker">{{ $t('enable-serviceworker') }}<template #desc>{{ $t('serviceworker-info') }}</template></ui-switch> | ||||
| 			<template v-if="enableServiceWorker"> | ||||
| 				<ui-info>{{ $t('vapid-info') }}<br><code>npm i web-push -g<br>web-push generate-vapid-keys</code></ui-info> | ||||
| 				<ui-horizon-group inputs class="fit-bottom"> | ||||
| 					<ui-input v-model="swPublicKey" :disabled="!enableServiceWorker"><template #icon><fa icon="key"/></template>{{ $t('vapid-publickey') }}</ui-input> | ||||
| 					<ui-input v-model="swPrivateKey" :disabled="!enableServiceWorker"><template #icon><fa icon="key"/></template>{{ $t('vapid-privatekey') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 			</template> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<template #title><fa :icon="faShieldAlt"/> {{ $t('recaptcha-config') }}</template> | ||||
| 		<section :class="enableRecaptcha ? 'fit-bottom' : ''"> | ||||
| 			<ui-switch v-model="enableRecaptcha">{{ $t('enable-recaptcha') }}</ui-switch> | ||||
| 			<template v-if="enableRecaptcha"> | ||||
| 				<ui-info>{{ $t('recaptcha-info') }}</ui-info> | ||||
| 				<ui-horizon-group inputs> | ||||
| 					<ui-input v-model="recaptchaSiteKey" :disabled="!enableRecaptcha"><template #icon><fa icon="key"/></template>{{ $t('recaptcha-site-key') }}</ui-input> | ||||
| 					<ui-input v-model="recaptchaSecretKey" :disabled="!enableRecaptcha"><template #icon><fa icon="key"/></template>{{ $t('recaptcha-secret-key') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 			</template> | ||||
| 		</section> | ||||
| 		<section v-if="enableRecaptcha && recaptchaSiteKey"> | ||||
| 			<header>{{ $t('recaptcha-preview') }}</header> | ||||
| 			<div ref="recaptcha" style="margin: 16px 0 0 0;" :key="recaptchaSiteKey"></div> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<template #title><fa :icon="faShieldAlt"/> {{ $t('external-service-integration-config') }}</template> | ||||
| 		<section> | ||||
| 			<header><fa :icon="['fab', 'twitter']"/> {{ $t('twitter-integration-config') }}</header> | ||||
| 			<ui-switch v-model="enableTwitterIntegration">{{ $t('enable-twitter-integration') }}</ui-switch> | ||||
| 			<ui-horizon-group> | ||||
| 				<ui-input v-model="twitterConsumerKey" :disabled="!enableTwitterIntegration"><i slot="icon"><fa icon="key"/></i>{{ $t('twitter-integration-consumer-key') }}</ui-input> | ||||
| 				<ui-input v-model="twitterConsumerSecret" :disabled="!enableTwitterIntegration"><i slot="icon"><fa icon="key"/></i>{{ $t('twitter-integration-consumer-secret') }}</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 			<ui-info>{{ $t('twitter-integration-info', { url: `${url}/api/tw/cb` }) }}</ui-info> | ||||
| 			<ui-button @click="updateMeta">{{ $t('save') }}</ui-button> | ||||
| 			<template v-if="enableTwitterIntegration"> | ||||
| 				<ui-horizon-group> | ||||
| 					<ui-input v-model="twitterConsumerKey" :disabled="!enableTwitterIntegration"><template #icon><fa icon="key"/></template>{{ $t('twitter-integration-consumer-key') }}</ui-input> | ||||
| 					<ui-input v-model="twitterConsumerSecret" :disabled="!enableTwitterIntegration"><template #icon><fa icon="key"/></template>{{ $t('twitter-integration-consumer-secret') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-info>{{ $t('twitter-integration-info', { url: `${url}/api/tw/cb` }) }}</ui-info> | ||||
| 			</template> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="['fab', 'github']"/> {{ $t('github-integration-config') }}</div> | ||||
| 		<section> | ||||
| 			<header><fa :icon="['fab', 'github']"/> {{ $t('github-integration-config') }}</header> | ||||
| 			<ui-switch v-model="enableGithubIntegration">{{ $t('enable-github-integration') }}</ui-switch> | ||||
| 			<ui-horizon-group> | ||||
| 				<ui-input v-model="githubClientId" :disabled="!enableGithubIntegration"><i slot="icon"><fa icon="key"/></i>{{ $t('github-integration-client-id') }}</ui-input> | ||||
| 				<ui-input v-model="githubClientSecret" :disabled="!enableGithubIntegration"><i slot="icon"><fa icon="key"/></i>{{ $t('github-integration-client-secret') }}</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 			<ui-info>{{ $t('github-integration-info', { url: `${url}/api/gh/cb` }) }}</ui-info> | ||||
| 			<ui-button @click="updateMeta">{{ $t('save') }}</ui-button> | ||||
| 			<template v-if="enableGithubIntegration"> | ||||
| 				<ui-horizon-group> | ||||
| 					<ui-input v-model="githubClientId" :disabled="!enableGithubIntegration"><template #icon><fa icon="key"/></template>{{ $t('github-integration-client-id') }}</ui-input> | ||||
| 					<ui-input v-model="githubClientSecret" :disabled="!enableGithubIntegration"><template #icon><fa icon="key"/></template>{{ $t('github-integration-client-secret') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-info>{{ $t('github-integration-info', { url: `${url}/api/gh/cb` }) }}</ui-info> | ||||
| 			</template> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<header><fa :icon="['fab', 'discord']"/> {{ $t('discord-integration-config') }}</header> | ||||
| 			<ui-switch v-model="enableDiscordIntegration">{{ $t('enable-discord-integration') }}</ui-switch> | ||||
| 			<template v-if="enableDiscordIntegration"> | ||||
| 				<ui-horizon-group> | ||||
| 					<ui-input v-model="discordClientId" :disabled="!enableDiscordIntegration"><template #icon><fa icon="key"/></template>{{ $t('discord-integration-client-id') }}</ui-input> | ||||
| 					<ui-input v-model="discordClientSecret" :disabled="!enableDiscordIntegration"><template #icon><fa icon="key"/></template>{{ $t('discord-integration-client-secret') }}</ui-input> | ||||
| 				</ui-horizon-group> | ||||
| 				<ui-info>{{ $t('discord-integration-info', { url: `${url}/api/dc/cb` }) }}</ui-info> | ||||
| 			</template> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="['fab', 'discord']"/> {{ $t('discord-integration-config') }}</div> | ||||
| 		<section> | ||||
| 			<ui-switch v-model="enableDiscordIntegration">{{ $t('enable-discord-integration') }}</ui-switch> | ||||
| 			<ui-horizon-group> | ||||
| 				<ui-input v-model="discordClientId" :disabled="!enableDiscordIntegration"><i slot="icon"><fa icon="key"/></i>{{ $t('discord-integration-client-id') }}</ui-input> | ||||
| 				<ui-input v-model="discordClientSecret" :disabled="!enableDiscordIntegration"><i slot="icon"><fa icon="key"/></i>{{ $t('discord-integration-client-secret') }}</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 			<ui-info>{{ $t('discord-integration-info', { url: `${url}/api/dc/cb` }) }}</ui-info> | ||||
| 			<ui-button @click="updateMeta">{{ $t('save') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
| 	<details> | ||||
| 		<summary style="color:var(--text);">{{ $t('advanced-config') }}</summary> | ||||
|  | ||||
| 		<ui-card> | ||||
| 			<template #title><fa :icon="faHashtag"/> {{ $t('hidden-tags') }}</template> | ||||
| 			<section class="fit-top"> | ||||
| 				<ui-textarea v-model="hiddenTags"> | ||||
| 					<template #desc>{{ $t('hidden-tags-info') }}</template> | ||||
| 				</ui-textarea> | ||||
| 				<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 			</section> | ||||
| 		</ui-card> | ||||
|  | ||||
| 		<ui-card> | ||||
| 			<template #title>summaly Proxy</template> | ||||
| 			<section class="fit-top fit-bottom"> | ||||
| 				<ui-input v-model="summalyProxy">URL</ui-input> | ||||
| 			</section> | ||||
| 			<section> | ||||
| 				<ui-button @click="updateMeta"><fa :icon="faSave"/> {{ $t('save') }}</ui-button> | ||||
| 			</section> | ||||
| 		</ui-card> | ||||
| 	</details> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| @@ -138,8 +245,8 @@ import Vue from 'vue'; | ||||
| import i18n from '../../i18n'; | ||||
| import { url, host } from '../../config'; | ||||
| import { toUnicode } from 'punycode'; | ||||
| import { faHeadset, faShieldAlt, faGhost, faUserPlus, faBolt } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faEnvelope as farEnvelope } from '@fortawesome/free-regular-svg-icons'; | ||||
| import { faHeadset, faShieldAlt, faGhost, faUserPlus, faBolt, faThumbtack, faPencilAlt, faHashtag } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faEnvelope as farEnvelope, faSave } from '@fortawesome/free-regular-svg-icons'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	i18n: i18n('admin/views/instance.vue'), | ||||
| @@ -150,12 +257,18 @@ export default Vue.extend({ | ||||
| 			host: toUnicode(host), | ||||
| 			maintainerName: null, | ||||
| 			maintainerEmail: null, | ||||
| 			ToSUrl: null, | ||||
| 			repositoryUrl: "https://github.com/syuilo/misskey", | ||||
| 			feedbackUrl: null, | ||||
| 			disableRegistration: false, | ||||
| 			disableLocalTimeline: false, | ||||
| 			disableGlobalTimeline: false, | ||||
| 			enableEmojiReaction: true, | ||||
| 			useStarForReactionFallback: false, | ||||
| 			mascotImageUrl: null, | ||||
| 			bannerUrl: null, | ||||
| 			errorImageUrl: null, | ||||
| 			iconUrl: null, | ||||
| 			name: null, | ||||
| 			description: null, | ||||
| 			languages: null, | ||||
| @@ -176,10 +289,6 @@ export default Vue.extend({ | ||||
| 			discordClientId: null, | ||||
| 			discordClientSecret: null, | ||||
| 			proxyAccount: null, | ||||
| 			inviteCode: null, | ||||
| 			enableExternalUserRecommendation: false, | ||||
| 			externalUserRecommendationEngine: null, | ||||
| 			externalUserRecommendationTimeout: null, | ||||
| 			summalyProxy: null, | ||||
| 			enableEmail: false, | ||||
| 			email: null, | ||||
| @@ -188,23 +297,42 @@ export default Vue.extend({ | ||||
| 			smtpPort: null, | ||||
| 			smtpUser: null, | ||||
| 			smtpPass: null, | ||||
| 			smtpAuth: false, | ||||
| 			enableServiceWorker: false, | ||||
| 			swPublicKey: null, | ||||
| 			swPrivateKey: null, | ||||
| 			faHeadset, faShieldAlt, faGhost, faUserPlus, farEnvelope, faBolt | ||||
| 			pinnedUsers: '', | ||||
| 			hiddenTags: '', | ||||
| 			useObjectStorage: false, | ||||
| 			objectStorageBaseUrl: null, | ||||
| 			objectStorageBucket: null, | ||||
| 			objectStoragePrefix: null, | ||||
| 			objectStorageEndpoint: null, | ||||
| 			objectStorageRegion: null, | ||||
| 			objectStoragePort: null, | ||||
| 			objectStorageAccessKey: null, | ||||
| 			objectStorageSecretKey: null, | ||||
| 			objectStorageUseSSL: false, | ||||
| 			faHeadset, faShieldAlt, faGhost, faUserPlus, farEnvelope, faBolt, faThumbtack, faPencilAlt, faSave, faHashtag | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	created() { | ||||
| 		this.$root.getMeta().then(meta => { | ||||
| 			this.maintainerName = meta.maintainer.name; | ||||
| 			this.maintainerEmail = meta.maintainer.email; | ||||
| 		this.$root.getMeta(true).then(meta => { | ||||
| 			this.maintainerName = meta.maintainerName; | ||||
| 			this.maintainerEmail = meta.maintainerEmail; | ||||
| 			this.ToSUrl = meta.ToSUrl; | ||||
| 			this.repositoryUrl = meta.repositoryUrl; | ||||
| 			this.feedbackUrl = meta.feedbackUrl; | ||||
| 			this.disableRegistration = meta.disableRegistration; | ||||
| 			this.disableLocalTimeline = meta.disableLocalTimeline; | ||||
| 			this.disableGlobalTimeline = meta.disableGlobalTimeline; | ||||
| 			this.enableEmojiReaction = meta.enableEmojiReaction; | ||||
| 			this.useStarForReactionFallback = meta.useStarForReactionFallback; | ||||
| 			this.mascotImageUrl = meta.mascotImageUrl; | ||||
| 			this.bannerUrl = meta.bannerUrl; | ||||
| 			this.errorImageUrl = meta.errorImageUrl; | ||||
| 			this.iconUrl = meta.iconUrl; | ||||
| 			this.name = meta.name; | ||||
| 			this.description = meta.description; | ||||
| 			this.languages = meta.langs.join(' '); | ||||
| @@ -225,9 +353,6 @@ export default Vue.extend({ | ||||
| 			this.enableDiscordIntegration = meta.enableDiscordIntegration; | ||||
| 			this.discordClientId = meta.discordClientId; | ||||
| 			this.discordClientSecret = meta.discordClientSecret; | ||||
| 			this.enableExternalUserRecommendation = meta.enableExternalUserRecommendation; | ||||
| 			this.externalUserRecommendationEngine = meta.externalUserRecommendationEngine; | ||||
| 			this.externalUserRecommendationTimeout = meta.externalUserRecommendationTimeout; | ||||
| 			this.summalyProxy = meta.summalyProxy; | ||||
| 			this.enableEmail = meta.enableEmail; | ||||
| 			this.email = meta.email; | ||||
| @@ -236,16 +361,60 @@ export default Vue.extend({ | ||||
| 			this.smtpPort = meta.smtpPort; | ||||
| 			this.smtpUser = meta.smtpUser; | ||||
| 			this.smtpPass = meta.smtpPass; | ||||
| 			this.smtpAuth = meta.smtpUser != null && meta.smtpUser !== ''; | ||||
| 			this.enableServiceWorker = meta.enableServiceWorker; | ||||
| 			this.swPublicKey = meta.swPublickey; | ||||
| 			this.swPrivateKey = meta.swPrivateKey; | ||||
| 			this.pinnedUsers = meta.pinnedUsers.join('\n'); | ||||
| 			this.hiddenTags = meta.hiddenTags.join('\n'); | ||||
| 			this.useObjectStorage = meta.useObjectStorage; | ||||
| 			this.objectStorageBaseUrl = meta.objectStorageBaseUrl; | ||||
| 			this.objectStorageBucket = meta.objectStorageBucket; | ||||
| 			this.objectStoragePrefix = meta.objectStoragePrefix; | ||||
| 			this.objectStorageEndpoint = meta.objectStorageEndpoint; | ||||
| 			this.objectStorageRegion = meta.objectStorageRegion; | ||||
| 			this.objectStoragePort = meta.objectStoragePort; | ||||
| 			this.objectStorageAccessKey = meta.objectStorageAccessKey; | ||||
| 			this.objectStorageSecretKey = meta.objectStorageSecretKey; | ||||
| 			this.objectStorageUseSSL = meta.objectStorageUseSSL; | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		const renderRecaptchaPreview = () => { | ||||
| 			if (!(window as any).grecaptcha) return; | ||||
| 			if (!this.$refs.recaptcha) return; | ||||
| 			if (!this.recaptchaSiteKey) return; | ||||
| 			(window as any).grecaptcha.render(this.$refs.recaptcha, { | ||||
| 				sitekey: this.recaptchaSiteKey | ||||
| 			}); | ||||
| 		}; | ||||
|  | ||||
| 		window.onRecaotchaLoad = () => { | ||||
| 			renderRecaptchaPreview(); | ||||
| 		}; | ||||
|  | ||||
| 		const head = document.getElementsByTagName('head')[0]; | ||||
| 		const script = document.createElement('script'); | ||||
| 		script.setAttribute('src', 'https://www.google.com/recaptcha/api.js?onload=onRecaotchaLoad'); | ||||
| 		head.appendChild(script); | ||||
|  | ||||
| 		this.$watch('enableRecaptcha', () => { | ||||
| 			renderRecaptchaPreview(); | ||||
| 		}); | ||||
|  | ||||
| 		this.$watch('recaptchaSiteKey', () => { | ||||
| 			renderRecaptchaPreview(); | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		invite() { | ||||
| 			this.$root.api('admin/invite').then(x => { | ||||
| 				this.inviteCode = x.code; | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'info', | ||||
| 					text: x.code | ||||
| 				}); | ||||
| 			}).catch(e => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'error', | ||||
| @@ -258,12 +427,18 @@ export default Vue.extend({ | ||||
| 			this.$root.api('admin/update-meta', { | ||||
| 				maintainerName: this.maintainerName, | ||||
| 				maintainerEmail: this.maintainerEmail, | ||||
| 				ToSUrl: this.ToSUrl, | ||||
| 				repositoryUrl: this.repositoryUrl, | ||||
| 				feedbackUrl: this.feedbackUrl, | ||||
| 				disableRegistration: this.disableRegistration, | ||||
| 				disableLocalTimeline: this.disableLocalTimeline, | ||||
| 				disableGlobalTimeline: this.disableGlobalTimeline, | ||||
| 				enableEmojiReaction: this.enableEmojiReaction, | ||||
| 				useStarForReactionFallback: this.useStarForReactionFallback, | ||||
| 				mascotImageUrl: this.mascotImageUrl, | ||||
| 				bannerUrl: this.bannerUrl, | ||||
| 				errorImageUrl: this.errorImageUrl, | ||||
| 				iconUrl: this.iconUrl, | ||||
| 				name: this.name, | ||||
| 				description: this.description, | ||||
| 				langs: this.languages.split(' '), | ||||
| @@ -284,20 +459,29 @@ export default Vue.extend({ | ||||
| 				enableDiscordIntegration: this.enableDiscordIntegration, | ||||
| 				discordClientId: this.discordClientId, | ||||
| 				discordClientSecret: this.discordClientSecret, | ||||
| 				enableExternalUserRecommendation: this.enableExternalUserRecommendation, | ||||
| 				externalUserRecommendationEngine: this.externalUserRecommendationEngine, | ||||
| 				externalUserRecommendationTimeout: parseInt(this.externalUserRecommendationTimeout, 10), | ||||
| 				summalyProxy: this.summalyProxy, | ||||
| 				enableEmail: this.enableEmail, | ||||
| 				email: this.email, | ||||
| 				smtpSecure: this.smtpSecure, | ||||
| 				smtpHost: this.smtpHost, | ||||
| 				smtpPort: parseInt(this.smtpPort, 10), | ||||
| 				smtpUser: this.smtpUser, | ||||
| 				smtpPass: this.smtpPass, | ||||
| 				smtpUser: this.smtpAuth ? this.smtpUser : '', | ||||
| 				smtpPass: this.smtpAuth ? this.smtpPass : '', | ||||
| 				enableServiceWorker: this.enableServiceWorker, | ||||
| 				swPublicKey: this.swPublicKey, | ||||
| 				swPrivateKey: this.swPrivateKey | ||||
| 				swPrivateKey: this.swPrivateKey, | ||||
| 				pinnedUsers: this.pinnedUsers.split('\n'), | ||||
| 				hiddenTags: this.hiddenTags.split('\n'), | ||||
| 				useObjectStorage: this.useObjectStorage, | ||||
| 				objectStorageBaseUrl: this.objectStorageBaseUrl ? this.objectStorageBaseUrl : null, | ||||
| 				objectStorageBucket: this.objectStorageBucket ? this.objectStorageBucket : null, | ||||
| 				objectStoragePrefix: this.objectStoragePrefix ? this.objectStoragePrefix : null, | ||||
| 				objectStorageEndpoint: this.objectStorageEndpoint ? this.objectStorageEndpoint : null, | ||||
| 				objectStorageRegion: this.objectStorageRegion ? this.objectStorageRegion : null, | ||||
| 				objectStoragePort: this.objectStoragePort ? this.objectStoragePort : null, | ||||
| 				objectStorageAccessKey: this.objectStorageAccessKey ? this.objectStorageAccessKey : null, | ||||
| 				objectStorageSecretKey: this.objectStorageSecretKey ? this.objectStorageSecretKey : null, | ||||
| 				objectStorageUseSSL: this.objectStorageUseSSL, | ||||
| 			}).then(() => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
|   | ||||
							
								
								
									
										108
									
								
								src/client/app/admin/views/logs.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								src/client/app/admin/views/logs.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,108 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<template #title><fa :icon="faStream"/> {{ $t('logs') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-horizon-group inputs> | ||||
| 				<ui-input v-model="domain" :debounce="true"> | ||||
| 					<span>{{ $t('domain') }}</span> | ||||
| 				</ui-input> | ||||
| 				<ui-select v-model="level"> | ||||
| 					<template #label>{{ $t('level') }}</template> | ||||
| 					<option value="all">{{ $t('levels.all') }}</option> | ||||
| 					<option value="info">{{ $t('levels.info') }}</option> | ||||
| 					<option value="success">{{ $t('levels.success') }}</option> | ||||
| 					<option value="warning">{{ $t('levels.warning') }}</option> | ||||
| 					<option value="error">{{ $t('levels.error') }}</option> | ||||
| 					<option value="debug">{{ $t('levels.debug') }}</option> | ||||
| 				</ui-select> | ||||
| 			</ui-horizon-group> | ||||
|  | ||||
| 			<div class="nqjzuvev"> | ||||
| 				<code v-for="log in logs" :key="log.id" :class="log.level"> | ||||
| 					<details> | ||||
| 						<summary><mk-time :time="log.createdAt"/> [{{ log.domain.join('.') }}] {{ log.message }}</summary> | ||||
| 						<vue-json-pretty v-if="log.data" :data="log.data"></vue-json-pretty> | ||||
| 					</details> | ||||
| 				</code> | ||||
| 			</div> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import i18n from '../../i18n'; | ||||
| import { faStream } from '@fortawesome/free-solid-svg-icons'; | ||||
| import VueJsonPretty from 'vue-json-pretty'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	i18n: i18n('admin/views/logs.vue'), | ||||
|  | ||||
| 	components: { | ||||
| 		VueJsonPretty | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			logs: [], | ||||
| 			level: 'all', | ||||
| 			domain: '', | ||||
| 			faStream | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	watch: { | ||||
| 		level() { | ||||
| 			this.logs = []; | ||||
| 			this.fetch(); | ||||
| 		}, | ||||
|  | ||||
| 		domain() { | ||||
| 			this.logs = []; | ||||
| 			this.fetch(); | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		this.fetch(); | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		fetch() { | ||||
| 			this.$root.api('admin/logs', { | ||||
| 				level: this.level === 'all' ? null : this.level, | ||||
| 				domain: this.domain === '' ? null : this.domain, | ||||
| 				limit: 100 | ||||
| 			}).then(logs => { | ||||
| 				this.logs = logs.reverse(); | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .nqjzuvev | ||||
| 	padding 8px | ||||
| 	background #000 | ||||
| 	color #fff | ||||
| 	font-size 14px | ||||
|  | ||||
| 	> code | ||||
| 		display block | ||||
|  | ||||
| 		&.error | ||||
| 			color #f00 | ||||
|  | ||||
| 		&.warning | ||||
| 			color #ff0 | ||||
|  | ||||
| 		&.success | ||||
| 			color #0f0 | ||||
|  | ||||
| 		&.debug | ||||
| 			opacity 0.7 | ||||
|  | ||||
| </style> | ||||
| @@ -1,10 +1,10 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa icon="plus"/> {{ $t('add-moderator.title') }}</div> | ||||
| 		<template #title><fa icon="plus"/> {{ $t('add-moderator.title') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-input v-model="username" type="text"> | ||||
| 				<span slot="prefix">@</span> | ||||
| 				<template #prefix>@</template> | ||||
| 			</ui-input> | ||||
| 			<ui-horizon-group> | ||||
| 				<ui-button @click="add" :disabled="changing">{{ $t('add-moderator.add') }}</ui-button> | ||||
|   | ||||
| @@ -1,26 +1,261 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title">{{ $t('operation') }}</div> | ||||
| 		<template #title><fa :icon="faChartBar"/> {{ $t('title') }}</template> | ||||
| 		<section class="wptihjuy"> | ||||
| 			<header><fa :icon="faPaperPlane"/> Deliver</header> | ||||
| 			<ui-info warn v-if="latestStats && latestStats.deliver.waiting > 0">The queue is jammed.</ui-info> | ||||
| 			<ui-horizon-group inputs v-if="latestStats" class="fit-bottom"> | ||||
| 				<ui-input :value="latestStats.deliver.activeSincePrevTick | number" type="text" readonly> | ||||
| 					<span>Process</span> | ||||
| 					<template #prefix><fa :icon="fasPlayCircle"/></template> | ||||
| 					<template #suffix>jobs/tick</template> | ||||
| 				</ui-input> | ||||
| 				<ui-input :value="latestStats.deliver.active | number" type="text" readonly> | ||||
| 					<span>Active</span> | ||||
| 					<template #prefix><fa :icon="farPlayCircle"/></template> | ||||
| 					<template #suffix>jobs</template> | ||||
| 				</ui-input> | ||||
| 				<ui-input :value="latestStats.deliver.waiting | number" type="text" readonly> | ||||
| 					<span>Waiting</span> | ||||
| 					<template #prefix><fa :icon="faStopCircle"/></template> | ||||
| 					<template #suffix>jobs</template> | ||||
| 				</ui-input> | ||||
| 				<ui-input :value="latestStats.deliver.delayed | number" type="text" readonly> | ||||
| 					<span>Delayed</span> | ||||
| 					<template #prefix><fa :icon="faStopwatch"/></template> | ||||
| 					<template #suffix>jobs</template> | ||||
| 				</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 			<div ref="deliverChart" class="chart"></div> | ||||
| 		</section> | ||||
| 		<section class="wptihjuy"> | ||||
| 			<header><fa :icon="faInbox"/> Inbox</header> | ||||
| 			<ui-info warn v-if="latestStats && latestStats.inbox.waiting > 0">The queue is jammed.</ui-info> | ||||
| 			<ui-horizon-group inputs v-if="latestStats" class="fit-bottom"> | ||||
| 				<ui-input :value="latestStats.inbox.activeSincePrevTick | number" type="text" readonly> | ||||
| 					<span>Process</span> | ||||
| 					<template #prefix><fa :icon="fasPlayCircle"/></template> | ||||
| 					<template #suffix>jobs/tick</template> | ||||
| 				</ui-input> | ||||
| 				<ui-input :value="latestStats.inbox.active | number" type="text" readonly> | ||||
| 					<span>Active</span> | ||||
| 					<template #prefix><fa :icon="farPlayCircle"/></template> | ||||
| 					<template #suffix>jobs</template> | ||||
| 				</ui-input> | ||||
| 				<ui-input :value="latestStats.inbox.waiting | number" type="text" readonly> | ||||
| 					<span>Waiting</span> | ||||
| 					<template #prefix><fa :icon="faStopCircle"/></template> | ||||
| 					<template #suffix>jobs</template> | ||||
| 				</ui-input> | ||||
| 				<ui-input :value="latestStats.inbox.delayed | number" type="text" readonly> | ||||
| 					<span>Delayed</span> | ||||
| 					<template #prefix><fa :icon="faStopwatch"/></template> | ||||
| 					<template #suffix>jobs</template> | ||||
| 				</ui-input> | ||||
| 			</ui-horizon-group> | ||||
| 			<div ref="inboxChart" class="chart"></div> | ||||
| 		</section> | ||||
| 		<section> | ||||
| 			<ui-button @click="removeAllJobs">{{ $t('remove-all-jobs') }}</ui-button> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<template #title><fa :icon="faTasks"/> {{ $t('jobs') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-horizon-group inputs> | ||||
| 				<ui-select v-model="domain"> | ||||
| 					<template #label>{{ $t('queue') }}</template> | ||||
| 					<option value="deliver">{{ $t('domains.deliver') }}</option> | ||||
| 					<option value="inbox">{{ $t('domains.inbox') }}</option> | ||||
| 				</ui-select> | ||||
| 				<ui-select v-model="state"> | ||||
| 					<template #label>{{ $t('state') }}</template> | ||||
| 					<option value="delayed">{{ $t('states.delayed') }}</option> | ||||
| 				</ui-select> | ||||
| 			</ui-horizon-group> | ||||
| 			<sequential-entrance animation="entranceFromTop" delay="25"> | ||||
| 				<div class="xvvuvgsv" v-for="job in jobs"> | ||||
| 					<b>{{ job.id }}</b> | ||||
| 					<template v-if="domain === 'deliver'"> | ||||
| 						<span>{{ job.data.to }}</span> | ||||
| 					</template> | ||||
| 					<template v-if="domain === 'inbox'"> | ||||
| 						<span>{{ job.activity.id }}</span> | ||||
| 					</template> | ||||
| 				</div> | ||||
| 			</sequential-entrance> | ||||
| 			<ui-info v-if="jobs.length == jobsLimit">{{ $t('result-is-truncated', { n: jobsLimit }) }}</ui-info> | ||||
| 		</section> | ||||
| 	</ui-card> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import i18n from '../../i18n'; | ||||
| import ApexCharts from 'apexcharts'; | ||||
| import * as tinycolor from 'tinycolor2'; | ||||
| import { faTasks, faInbox, faStopwatch, faPlayCircle as fasPlayCircle } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faPaperPlane, faStopCircle, faPlayCircle as farPlayCircle, faChartBar } from '@fortawesome/free-regular-svg-icons'; | ||||
|  | ||||
| const limit = 200; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	i18n: i18n('admin/views/queue.vue'), | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			stats: [], | ||||
| 			deliverChart: null, | ||||
| 			inboxChart: null, | ||||
| 			jobs: [], | ||||
| 			jobsLimit: 50, | ||||
| 			domain: 'deliver', | ||||
| 			state: 'delayed', | ||||
| 			faTasks, faPaperPlane, faInbox, faStopwatch, faStopCircle, farPlayCircle, fasPlayCircle, faChartBar | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	computed: { | ||||
| 		latestStats(): any { | ||||
| 			return this.stats[this.stats.length - 1]; | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	watch: { | ||||
| 		stats(stats) { | ||||
| 			this.inboxChart.updateSeries([{ | ||||
| 				name: 'Process', | ||||
| 				type: 'area', | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.inbox.activeSincePrevTick })) | ||||
| 			}, { | ||||
| 				name: 'Active', | ||||
| 				type: 'area', | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.inbox.active })) | ||||
| 			}, { | ||||
| 				name: 'Waiting', | ||||
| 				type: 'line', | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.inbox.waiting })) | ||||
| 			}, { | ||||
| 				name: 'Delayed', | ||||
| 				type: 'line', | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.inbox.delayed })) | ||||
| 			}]); | ||||
| 			this.deliverChart.updateSeries([{ | ||||
| 				name: 'Process', | ||||
| 				type: 'area', | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.deliver.activeSincePrevTick })) | ||||
| 			}, { | ||||
| 				name: 'Active', | ||||
| 				type: 'area', | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.deliver.active })) | ||||
| 			}, { | ||||
| 				name: 'Waiting', | ||||
| 				type: 'line', | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.deliver.waiting })) | ||||
| 			}, { | ||||
| 				name: 'Delayed', | ||||
| 				type: 'line', | ||||
| 				data: stats.map((x, i) => ({ x: i, y: x.deliver.delayed })) | ||||
| 			}]); | ||||
| 		}, | ||||
|  | ||||
| 		domain() { | ||||
| 			this.jobs = []; | ||||
| 			this.fetchJobs(); | ||||
| 		}, | ||||
|  | ||||
| 		state() { | ||||
| 			this.jobs = []; | ||||
| 			this.fetchJobs(); | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		this.fetchJobs(); | ||||
|  | ||||
| 		const chartOpts = id => ({ | ||||
| 			chart: { | ||||
| 				id, | ||||
| 				group: 'queue', | ||||
| 				type: 'area', | ||||
| 				height: 200, | ||||
| 				animations: { | ||||
| 					dynamicAnimation: { | ||||
| 						enabled: false | ||||
| 					} | ||||
| 				}, | ||||
| 				toolbar: { | ||||
| 					show: false | ||||
| 				}, | ||||
| 				zoom: { | ||||
| 					enabled: false | ||||
| 				} | ||||
| 			}, | ||||
| 			dataLabels: { | ||||
| 				enabled: false | ||||
| 			}, | ||||
| 			grid: { | ||||
| 				clipMarkers: false, | ||||
| 				borderColor: 'rgba(0, 0, 0, 0.1)', | ||||
| 				xaxis: { | ||||
| 					lines: { | ||||
| 						show: true, | ||||
| 					} | ||||
| 				}, | ||||
| 			}, | ||||
| 			stroke: { | ||||
| 				curve: 'straight', | ||||
| 				width: 2 | ||||
| 			}, | ||||
| 			tooltip: { | ||||
| 				enabled: false | ||||
| 			}, | ||||
| 			legend: { | ||||
| 				labels: { | ||||
| 					colors: tinycolor(getComputedStyle(document.documentElement).getPropertyValue('--text')).toRgbString() | ||||
| 				}, | ||||
| 			}, | ||||
| 			series: [] as any, | ||||
| 			colors: ['#00E396', '#00BCD4', '#FFB300', '#e53935'], | ||||
| 			xaxis: { | ||||
| 				type: 'numeric', | ||||
| 				labels: { | ||||
| 					show: false | ||||
| 				}, | ||||
| 				tooltip: { | ||||
| 					enabled: false | ||||
| 				} | ||||
| 			}, | ||||
| 			yaxis: { | ||||
| 				show: false, | ||||
| 				min: 0, | ||||
| 			} | ||||
| 		}); | ||||
|  | ||||
| 		this.inboxChart = new ApexCharts(this.$refs.inboxChart, chartOpts('a')); | ||||
| 		this.deliverChart = new ApexCharts(this.$refs.deliverChart, chartOpts('b')); | ||||
|  | ||||
| 		this.inboxChart.render(); | ||||
| 		this.deliverChart.render(); | ||||
|  | ||||
| 		const connection = this.$root.stream.useSharedConnection('queueStats'); | ||||
| 		connection.on('stats', this.onStats); | ||||
| 		connection.on('statsLog', this.onStatsLog); | ||||
| 		connection.send('requestLog', { | ||||
| 			id: Math.random().toString().substr(2, 8), | ||||
| 			length: limit | ||||
| 		}); | ||||
|  | ||||
| 		this.$once('hook:beforeDestroy', () => { | ||||
| 			connection.dispose(); | ||||
| 			this.inboxChart.destroy(); | ||||
| 			this.deliverChart.destroy(); | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		async removeAllJobs() { | ||||
| 			const process = async () => { | ||||
| @@ -38,6 +273,39 @@ export default Vue.extend({ | ||||
| 				}); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		onStats(stats) { | ||||
| 			this.stats.push(stats); | ||||
| 			if (this.stats.length > limit) this.stats.shift(); | ||||
| 		}, | ||||
|  | ||||
| 		onStatsLog(statsLog) { | ||||
| 			for (const stats of statsLog.reverse()) { | ||||
| 				this.onStats(stats); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		fetchJobs() { | ||||
| 			this.$root.api('admin/queue/jobs', { | ||||
| 				domain: this.domain, | ||||
| 				state: this.state, | ||||
| 				limit: this.jobsLimit | ||||
| 			}).then(jobs => { | ||||
| 				this.jobs = jobs; | ||||
| 			}); | ||||
| 		}, | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .wptihjuy | ||||
| 	> .chart | ||||
| 		min-height 200px !important | ||||
| 		margin 0 -8px | ||||
|  | ||||
| .xvvuvgsv | ||||
| 	> b | ||||
| 		margin-right 16px | ||||
|  | ||||
| </style> | ||||
|   | ||||
| @@ -11,7 +11,6 @@ | ||||
| 			<span class="username">@{{ user | acct }}</span> | ||||
| 			<span class="is-admin" v-if="user.isAdmin">admin</span> | ||||
| 			<span class="is-moderator" v-if="user.isModerator">moderator</span> | ||||
| 			<span class="is-verified" v-if="user.isVerified" :title="$t('@.verified-user')"><fa icon="star"/></span> | ||||
| 			<span class="is-silenced" v-if="user.isSilenced" :title="$t('@.silenced-user')"><fa :icon="faMicrophoneSlash"/></span> | ||||
| 			<span class="is-suspended" v-if="user.isSuspended" :title="$t('@.suspended-user')"><fa :icon="faSnowflake"/></span> | ||||
| 		</header> | ||||
| @@ -77,7 +76,6 @@ export default Vue.extend({ | ||||
| 				background var(--noteHeaderAdminBg) | ||||
| 				color var(--noteHeaderAdminFg) | ||||
|  | ||||
| 			> .is-verified | ||||
| 			> .is-silenced | ||||
| 			> .is-suspended | ||||
| 				margin 0 0 0 .5em | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="faTerminal"/> {{ $t('operation') }}</div> | ||||
| 		<template #title><fa :icon="faTerminal"/> {{ $t('operation') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-input class="target" v-model="target" type="text" @enter="showUser"> | ||||
| 				<span>{{ $t('username-or-userid') }}</span> | ||||
| @@ -9,13 +9,10 @@ | ||||
| 			<ui-button @click="showUser"><fa :icon="faSearch"/> {{ $t('lookup') }}</ui-button> | ||||
|  | ||||
| 			<div class="user" v-if="user"> | ||||
| 				<x-user :user='user'/> | ||||
| 				<x-user :user="user"/> | ||||
| 				<div class="actions"> | ||||
| 					<ui-button v-if="user.host != null" @click="updateRemoteUser"><fa :icon="faSync"/> {{ $t('update-remote-user') }}</ui-button> | ||||
| 					<ui-button @click="resetPassword"><fa :icon="faKey"/> {{ $t('reset-password') }}</ui-button> | ||||
| 					<ui-horizon-group> | ||||
| 						<ui-button @click="verifyUser" :disabled="verifying"><fa :icon="faCertificate"/> {{ $t('verify') }}</ui-button> | ||||
| 						<ui-button @click="unverifyUser" :disabled="unverifying">{{ $t('unverify') }}</ui-button> | ||||
| 					</ui-horizon-group> | ||||
| 					<ui-horizon-group> | ||||
| 						<ui-button @click="silenceUser"><fa :icon="faMicrophoneSlash"/> {{ $t('make-silence') }}</ui-button> | ||||
| 						<ui-button @click="unsilenceUser">{{ $t('unmake-silence') }}</ui-button> | ||||
| @@ -24,7 +21,7 @@ | ||||
| 						<ui-button @click="suspendUser" :disabled="suspending"><fa :icon="faSnowflake"/> {{ $t('suspend') }}</ui-button> | ||||
| 						<ui-button @click="unsuspendUser" :disabled="unsuspending">{{ $t('unsuspend') }}</ui-button> | ||||
| 					</ui-horizon-group> | ||||
| 					<ui-button v-if="user.host != null" @click="updateRemoteUser"><fa :icon="faSync"/> {{ $t('update-remote-user') }}</ui-button> | ||||
| 					<ui-button @click="deleteAllFiles"><fa :icon="faTrashAlt"/> {{ $t('delete-all-files') }}</ui-button> | ||||
| 					<ui-textarea v-if="user" :value="user | json5" readonly tall style="margin-top:16px;"></ui-textarea> | ||||
| 				</div> | ||||
| 			</div> | ||||
| @@ -32,27 +29,26 @@ | ||||
| 	</ui-card> | ||||
|  | ||||
| 	<ui-card> | ||||
| 		<div slot="title"><fa :icon="faUsers"/> {{ $t('users.title') }}</div> | ||||
| 		<template #title><fa :icon="faUsers"/> {{ $t('users.title') }}</template> | ||||
| 		<section class="fit-top"> | ||||
| 			<ui-horizon-group inputs> | ||||
| 				<ui-select v-model="sort"> | ||||
| 					<span slot="label">{{ $t('users.sort.title') }}</span> | ||||
| 					<template #label>{{ $t('users.sort.title') }}</template> | ||||
| 					<option value="-createdAt">{{ $t('users.sort.createdAtAsc') }}</option> | ||||
| 					<option value="+createdAt">{{ $t('users.sort.createdAtDesc') }}</option> | ||||
| 					<option value="-updatedAt">{{ $t('users.sort.updatedAtAsc') }}</option> | ||||
| 					<option value="+updatedAt">{{ $t('users.sort.updatedAtDesc') }}</option> | ||||
| 				</ui-select> | ||||
| 				<ui-select v-model="state"> | ||||
| 					<span slot="label">{{ $t('users.state.title') }}</span> | ||||
| 					<template #label>{{ $t('users.state.title') }}</template> | ||||
| 					<option value="all">{{ $t('users.state.all') }}</option> | ||||
| 					<option value="admin">{{ $t('users.state.admin') }}</option> | ||||
| 					<option value="moderator">{{ $t('users.state.moderator') }}</option> | ||||
| 					<option value="verified">{{ $t('users.state.verified') }}</option> | ||||
| 					<option value="silenced">{{ $t('users.state.silenced') }}</option> | ||||
| 					<option value="suspended">{{ $t('users.state.suspended') }}</option> | ||||
| 				</ui-select> | ||||
| 				<ui-select v-model="origin"> | ||||
| 					<span slot="label">{{ $t('users.origin.title') }}</span> | ||||
| 					<template #label>{{ $t('users.origin.title') }}</template> | ||||
| 					<option value="combined">{{ $t('users.origin.combined') }}</option> | ||||
| 					<option value="local">{{ $t('users.origin.local') }}</option> | ||||
| 					<option value="remote">{{ $t('users.origin.remote') }}</option> | ||||
| @@ -71,8 +67,8 @@ | ||||
| import Vue from 'vue'; | ||||
| import i18n from '../../i18n'; | ||||
| import parseAcct from "../../../../misc/acct/parse"; | ||||
| import { faCertificate, faUsers, faTerminal, faSearch, faKey, faSync, faMicrophoneSlash } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faSnowflake } from '@fortawesome/free-regular-svg-icons'; | ||||
| import { faUsers, faTerminal, faSearch, faKey, faSync, faMicrophoneSlash } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faSnowflake, faTrashAlt } from '@fortawesome/free-regular-svg-icons'; | ||||
| import XUser from './users.user.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| @@ -84,8 +80,6 @@ export default Vue.extend({ | ||||
| 		return { | ||||
| 			user: null, | ||||
| 			target: null, | ||||
| 			verifying: false, | ||||
| 			unverifying: false, | ||||
| 			suspending: false, | ||||
| 			unsuspending: false, | ||||
| 			sort: '+createdAt', | ||||
| @@ -95,7 +89,7 @@ export default Vue.extend({ | ||||
| 			offset: 0, | ||||
| 			users: [], | ||||
| 			existMore: false, | ||||
| 			faTerminal, faCertificate, faUsers, faSnowflake, faSearch, faKey, faSync, faMicrophoneSlash | ||||
| 			faTerminal, faUsers, faSnowflake, faSearch, faKey, faSync, faMicrophoneSlash, faTrashAlt | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| @@ -165,7 +159,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 		/** 処理対象ユーザーの情報を更新する */ | ||||
| 		async refreshUser() { | ||||
| 			this.$root.api('admin/show-user', { userId: this.user._id }).then(info => { | ||||
| 			this.$root.api('admin/show-user', { userId: this.user.id }).then(info => { | ||||
| 				this.user = info; | ||||
| 			}); | ||||
| 		}, | ||||
| @@ -173,7 +167,7 @@ export default Vue.extend({ | ||||
| 		async resetPassword() { | ||||
| 			if (!await this.getConfirmed(this.$t('reset-password-confirm'))) return; | ||||
|  | ||||
| 			this.$root.api('admin/reset-password', { userId: this.user._id }).then(res => { | ||||
| 			this.$root.api('admin/reset-password', { userId: this.user.id }).then(res => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					text: this.$t('password-updated', { password: res.password }) | ||||
| @@ -181,59 +175,11 @@ export default Vue.extend({ | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		async verifyUser() { | ||||
| 			if (!await this.getConfirmed(this.$t('verify-confirm'))) return; | ||||
|  | ||||
| 			this.verifying = true; | ||||
|  | ||||
| 			const process = async () => { | ||||
| 				await this.$root.api('admin/verify-user', { userId: this.user._id }); | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					text: this.$t('verified') | ||||
| 				}); | ||||
| 			}; | ||||
|  | ||||
| 			await process().catch(e => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'error', | ||||
| 					text: e.toString() | ||||
| 				}); | ||||
| 			}); | ||||
|  | ||||
| 			this.verifying = false; | ||||
|  | ||||
| 			this.refreshUser(); | ||||
| 		}, | ||||
|  | ||||
| 		async unverifyUser() { | ||||
| 			if (!await this.getConfirmed(this.$t('unverify-confirm'))) return; | ||||
|  | ||||
| 			this.unverifying = true; | ||||
|  | ||||
| 			const process = async () => { | ||||
| 				await this.$root.api('admin/unverify-user', { userId: this.user._id }); | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					text: this.$t('unverified') | ||||
| 				}); | ||||
| 			}; | ||||
|  | ||||
| 			await process().catch(e => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'error', | ||||
| 					text: e.toString() | ||||
| 				}); | ||||
| 			}); | ||||
|  | ||||
| 			this.unverifying = false; | ||||
|  | ||||
| 			this.refreshUser(); | ||||
| 		}, | ||||
|  | ||||
| 		async silenceUser() { | ||||
| 			if (!await this.getConfirmed(this.$t('silence-confirm'))) return; | ||||
|  | ||||
| 			const process = async () => { | ||||
| 				await this.$root.api('admin/silence-user', { userId: this.user._id }); | ||||
| 				await this.$root.api('admin/silence-user', { userId: this.user.id }); | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					splash: true | ||||
| @@ -251,8 +197,10 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		async unsilenceUser() { | ||||
| 			if (!await this.getConfirmed(this.$t('unsilence-confirm'))) return; | ||||
|  | ||||
| 			const process = async () => { | ||||
| 				await this.$root.api('admin/unsilence-user', { userId: this.user._id }); | ||||
| 				await this.$root.api('admin/unsilence-user', { userId: this.user.id }); | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					splash: true | ||||
| @@ -275,7 +223,7 @@ export default Vue.extend({ | ||||
| 			this.suspending = true; | ||||
|  | ||||
| 			const process = async () => { | ||||
| 				await this.$root.api('admin/suspend-user', { userId: this.user._id }); | ||||
| 				await this.$root.api('admin/suspend-user', { userId: this.user.id }); | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					text: this.$t('suspended') | ||||
| @@ -300,7 +248,7 @@ export default Vue.extend({ | ||||
| 			this.unsuspending = true; | ||||
|  | ||||
| 			const process = async () => { | ||||
| 				await this.$root.api('admin/unsuspend-user', { userId: this.user._id }); | ||||
| 				await this.$root.api('admin/unsuspend-user', { userId: this.user.id }); | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					text: this.$t('unsuspended') | ||||
| @@ -320,7 +268,7 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		async updateRemoteUser() { | ||||
| 			this.$root.api('admin/update-remote-user', { userId: this.user._id }).then(res => { | ||||
| 			this.$root.api('admin/update-remote-user', { userId: this.user.id }).then(res => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					text: this.$t('remote-user-updated') | ||||
| @@ -330,6 +278,25 @@ export default Vue.extend({ | ||||
| 			this.refreshUser(); | ||||
| 		}, | ||||
|  | ||||
| 		async deleteAllFiles() { | ||||
| 			if (!await this.getConfirmed(this.$t('delete-all-files-confirm'))) return; | ||||
|  | ||||
| 			const process = async () => { | ||||
| 				await this.$root.api('admin/delete-all-files-of-a-user', { userId: this.user.id }); | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'success', | ||||
| 					splash: true | ||||
| 				}); | ||||
| 			}; | ||||
|  | ||||
| 			await process().catch(e => { | ||||
| 				this.$root.dialog({ | ||||
| 					type: 'error', | ||||
| 					text: e.toString() | ||||
| 				}); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		async getConfirmed(text: string): Promise<Boolean> { | ||||
| 			const confirm = await this.$root.dialog({ | ||||
| 				type: 'warning', | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user